diff options
Diffstat (limited to 'xorg-server/exa')
| -rw-r--r-- | xorg-server/exa/exa_accel.c | 2608 | ||||
| -rw-r--r-- | xorg-server/exa/exa_migration_classic.c | 1490 | ||||
| -rw-r--r-- | xorg-server/exa/exa_unaccel.c | 1496 | ||||
| -rw-r--r-- | xorg-server/exa/makefile | 14 | 
4 files changed, 2811 insertions, 2797 deletions
| diff --git a/xorg-server/exa/exa_accel.c b/xorg-server/exa/exa_accel.c index 4680c3709..00a97fdbf 100644 --- a/xorg-server/exa/exa_accel.c +++ b/xorg-server/exa/exa_accel.c @@ -1,1304 +1,1304 @@ -/* - * Copyright © 2001 Keith Packard - * - * Partly based on code that is Copyright © The XFree86 Project Inc. - * - * 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. - * - * Authors: - *    Eric Anholt <eric@anholt.net> - *    Michel Dänzer <michel@tungstengraphics.com> - * - */ - -#ifdef HAVE_DIX_CONFIG_H -#include <dix-config.h> -#endif -#include "exa_priv.h" -#include <X11/fonts/fontstruct.h> -#include "dixfontstr.h" -#include "exa.h" - -static void -exaFillSpans(DrawablePtr pDrawable, GCPtr pGC, int n, -	     DDXPointPtr ppt, int *pwidth, int fSorted) -{ -    ScreenPtr	    pScreen = pDrawable->pScreen; -    ExaScreenPriv (pScreen); -    RegionPtr	    pClip = fbGetCompositeClip(pGC); -    PixmapPtr	    pPixmap = exaGetDrawablePixmap (pDrawable); -    ExaPixmapPriv (pPixmap); -    BoxPtr	    pextent, pbox; -    int		    nbox; -    int		    extentX1, extentX2, extentY1, extentY2; -    int		    fullX1, fullX2, fullY1; -    int		    partX1, partX2; -    int		    off_x, off_y; - -    if (pExaScr->fallback_counter || -	pExaScr->swappedOut || -	pGC->fillStyle != FillSolid || -	pExaPixmap->accel_blocked) -    { -	ExaCheckFillSpans (pDrawable, pGC, n, ppt, pwidth, fSorted); -	return; -    } - -    if (pExaScr->do_migration) { -	ExaMigrationRec pixmaps[1]; - -	pixmaps[0].as_dst = TRUE; -	pixmaps[0].as_src = FALSE; -	pixmaps[0].pPix = pPixmap; -	pixmaps[0].pReg = NULL; - -	exaDoMigration (pixmaps, 1, TRUE); -    } - -    if (!(pPixmap = exaGetOffscreenPixmap (pDrawable, &off_x, &off_y)) || -	!(*pExaScr->info->PrepareSolid) (pPixmap, -					 pGC->alu, -					 pGC->planemask, -					 pGC->fgPixel)) -    { -	ExaCheckFillSpans (pDrawable, pGC, n, ppt, pwidth, fSorted); -	return; -    } - -    pextent = REGION_EXTENTS(pGC->pScreen, pClip); -    extentX1 = pextent->x1; -    extentY1 = pextent->y1; -    extentX2 = pextent->x2; -    extentY2 = pextent->y2; -    while (n--) -    { -	fullX1 = ppt->x; -	fullY1 = ppt->y; -	fullX2 = fullX1 + (int) *pwidth; -	ppt++; -	pwidth++; - -	if (fullY1 < extentY1 || extentY2 <= fullY1) -	    continue; - -	if (fullX1 < extentX1) -	    fullX1 = extentX1; - -	if (fullX2 > extentX2) -	    fullX2 = extentX2; - -	if (fullX1 >= fullX2) -	    continue; - -	nbox = REGION_NUM_RECTS (pClip); -	if (nbox == 1) -	{ -	    (*pExaScr->info->Solid) (pPixmap, -				     fullX1 + off_x, fullY1 + off_y, -				     fullX2 + off_x, fullY1 + 1 + off_y); -	} -	else -	{ -	    pbox = REGION_RECTS(pClip); -	    while(nbox--) -	    { -		if (pbox->y1 <= fullY1 && fullY1 < pbox->y2) -		{ -		    partX1 = pbox->x1; -		    if (partX1 < fullX1) -			partX1 = fullX1; -		    partX2 = pbox->x2; -		    if (partX2 > fullX2) -			partX2 = fullX2; -		    if (partX2 > partX1) { -			(*pExaScr->info->Solid) (pPixmap, -						 partX1 + off_x, fullY1 + off_y, -						 partX2 + off_x, fullY1 + 1 + off_y); -		    } -		} -		pbox++; -	    } -	} -    } -    (*pExaScr->info->DoneSolid) (pPixmap); -    exaMarkSync(pScreen); -} - -static Bool -exaDoPutImage (DrawablePtr pDrawable, GCPtr pGC, int depth, int x, int y, -	       int w, int h, int format, char *bits, int src_stride) -{ -    ExaScreenPriv (pDrawable->pScreen); -    PixmapPtr pPix = exaGetDrawablePixmap (pDrawable); -    ExaPixmapPriv(pPix); -    RegionPtr pClip; -    BoxPtr pbox; -    int nbox; -    int xoff, yoff; -    int bpp = pDrawable->bitsPerPixel; -    Bool ret = TRUE; - -    if (pExaScr->fallback_counter || pExaPixmap->accel_blocked || !pExaScr->info->UploadToScreen) -	return FALSE; - -    /* If there's a system copy, we want to save the result there */ -    if (pExaPixmap->pDamage) -	return FALSE; - -    /* Don't bother with under 8bpp, XYPixmaps. */ -    if (format != ZPixmap || bpp < 8) -	return FALSE; - -    /* Only accelerate copies: no rop or planemask. */ -    if (!EXA_PM_IS_SOLID(pDrawable, pGC->planemask) || pGC->alu != GXcopy) -	return FALSE; - -    if (pExaScr->swappedOut) -	return FALSE; - -    if (pExaScr->do_migration) { -	ExaMigrationRec pixmaps[1]; - -	pixmaps[0].as_dst = TRUE; -	pixmaps[0].as_src = FALSE; -	pixmaps[0].pPix = pPix; -	pixmaps[0].pReg = DamagePendingRegion(pExaPixmap->pDamage); - -	exaDoMigration (pixmaps, 1, TRUE); -    } - -    pPix = exaGetOffscreenPixmap (pDrawable, &xoff, &yoff); - -    if (!pPix) -	return FALSE; - -    x += pDrawable->x; -    y += pDrawable->y; - -    pClip = fbGetCompositeClip(pGC); -    for (nbox = REGION_NUM_RECTS(pClip), -	 pbox = REGION_RECTS(pClip); -	 nbox--; -	 pbox++) -    { -	int x1 = x; -	int y1 = y; -	int x2 = x + w; -	int y2 = y + h; -	char *src; -	Bool ok; - -	if (x1 < pbox->x1) -	    x1 = pbox->x1; -	if (y1 < pbox->y1) -	    y1 = pbox->y1; -	if (x2 > pbox->x2) -	    x2 = pbox->x2; -	if (y2 > pbox->y2) -	    y2 = pbox->y2; -	if (x1 >= x2 || y1 >= y2) -	    continue; - -	src = bits + (y1 - y) * src_stride + (x1 - x) * (bpp / 8); -	ok = pExaScr->info->UploadToScreen(pPix, x1 + xoff, y1 + yoff, -					   x2 - x1, y2 - y1, src, src_stride); -	/* We have to fall back completely, and ignore what has already been completed. -	 * Messing with the fb layer directly like we used to is completely unacceptable. -	 */ -	if (!ok) { -	    ret = FALSE; -	    break; -	} -    } - -    if (ret) -	exaMarkSync(pDrawable->pScreen); - -    return ret; -} - -static void -exaPutImage (DrawablePtr pDrawable, GCPtr pGC, int depth, int x, int y, -	     int w, int h, int leftPad, int format, char *bits) -{ -    if (!exaDoPutImage(pDrawable, pGC, depth, x, y, w, h, format, bits, -		       PixmapBytePad(w, pDrawable->depth))) -	ExaCheckPutImage(pDrawable, pGC, depth, x, y, w, h, leftPad, format, -			 bits); -} - -static Bool inline -exaCopyNtoNTwoDir (DrawablePtr pSrcDrawable, DrawablePtr pDstDrawable, -		   GCPtr pGC, BoxPtr pbox, int nbox, int dx, int dy) -{ -    ExaScreenPriv (pDstDrawable->pScreen); -    PixmapPtr pSrcPixmap, pDstPixmap; -    int src_off_x, src_off_y, dst_off_x, dst_off_y; -    int dirsetup; - -    /* Need to get both pixmaps to call the driver routines */ -    pSrcPixmap = exaGetOffscreenPixmap (pSrcDrawable, &src_off_x, &src_off_y); -    pDstPixmap = exaGetOffscreenPixmap (pDstDrawable, &dst_off_x, &dst_off_y); -    if (!pSrcPixmap || !pDstPixmap) -	return FALSE; - -    /* -     * Now the case of a chip that only supports xdir = ydir = 1 or -     * xdir = ydir = -1, but we have xdir != ydir. -     */ -    dirsetup = 0;	/* No direction set up yet. */ -    for (; nbox; pbox++, nbox--) { -	if (dx >= 0 && (src_off_y + pbox->y1 + dy) != pbox->y1) { -	    /* Do a xdir = ydir = -1 blit instead. */ -	    if (dirsetup != -1) { -		if (dirsetup != 0) -		    pExaScr->info->DoneCopy(pDstPixmap); -		dirsetup = -1; -		if (!(*pExaScr->info->PrepareCopy)(pSrcPixmap, -						   pDstPixmap, -						   -1, -1, -						   pGC ? pGC->alu : GXcopy, -						   pGC ? pGC->planemask : -							 FB_ALLONES)) -		    return FALSE; -	    } -	    (*pExaScr->info->Copy)(pDstPixmap, -				   src_off_x + pbox->x1 + dx, -				   src_off_y + pbox->y1 + dy, -				   dst_off_x + pbox->x1, -				   dst_off_y + pbox->y1, -				   pbox->x2 - pbox->x1, -				   pbox->y2 - pbox->y1); -	} else if (dx < 0 && (src_off_y + pbox->y1 + dy) != pbox->y1) { -	    /* Do a xdir = ydir = 1 blit instead. */ -	    if (dirsetup != 1) { -		if (dirsetup != 0) -		    pExaScr->info->DoneCopy(pDstPixmap); -		dirsetup = 1; -		if (!(*pExaScr->info->PrepareCopy)(pSrcPixmap, -						   pDstPixmap, -						   1, 1, -						   pGC ? pGC->alu : GXcopy, -						   pGC ? pGC->planemask : -							 FB_ALLONES)) -		    return FALSE; -	    } -	    (*pExaScr->info->Copy)(pDstPixmap, -				   src_off_x + pbox->x1 + dx, -				   src_off_y + pbox->y1 + dy, -				   dst_off_x + pbox->x1, -				   dst_off_y + pbox->y1, -				   pbox->x2 - pbox->x1, -				   pbox->y2 - pbox->y1); -	} else if (dx >= 0) { -	    /* -	     * xdir = 1, ydir = -1. -	     * Perform line-by-line xdir = ydir = 1 blits, going up. -	     */ -	    int i; -	    if (dirsetup != 1) { -		if (dirsetup != 0) -		    pExaScr->info->DoneCopy(pDstPixmap); -		dirsetup = 1; -		if (!(*pExaScr->info->PrepareCopy)(pSrcPixmap, -						   pDstPixmap, -						   1, 1, -						   pGC ? pGC->alu : GXcopy, -						   pGC ? pGC->planemask : -							 FB_ALLONES)) -		    return FALSE; -	    } -	    for (i = pbox->y2 - pbox->y1 - 1; i >= 0; i--) -		(*pExaScr->info->Copy)(pDstPixmap, -				       src_off_x + pbox->x1 + dx, -				       src_off_y + pbox->y1 + dy + i, -				       dst_off_x + pbox->x1, -				       dst_off_y + pbox->y1 + i, -				       pbox->x2 - pbox->x1, 1); -	} else { -	    /* -	     * xdir = -1, ydir = 1. -	     * Perform line-by-line xdir = ydir = -1 blits, going down. -	     */ -	    int i; -	    if (dirsetup != -1) { -		if (dirsetup != 0) -		    pExaScr->info->DoneCopy(pDstPixmap); -		dirsetup = -1; -		if (!(*pExaScr->info->PrepareCopy)(pSrcPixmap, -						   pDstPixmap, -						   -1, -1, -						   pGC ? pGC->alu : GXcopy, -						   pGC ? pGC->planemask : -							 FB_ALLONES)) -		    return FALSE; -	    } -	    for (i = 0; i < pbox->y2 - pbox->y1; i++) -		(*pExaScr->info->Copy)(pDstPixmap, -				       src_off_x + pbox->x1 + dx, -				       src_off_y + pbox->y1 + dy + i, -				       dst_off_x + pbox->x1, -				       dst_off_y + pbox->y1 + i, -				       pbox->x2 - pbox->x1, 1); -	} -    } -    if (dirsetup != 0) -	pExaScr->info->DoneCopy(pDstPixmap); -    exaMarkSync(pDstDrawable->pScreen); -    return TRUE; -} - -Bool -exaHWCopyNtoN (DrawablePtr    pSrcDrawable, -	     DrawablePtr    pDstDrawable, -	     GCPtr	    pGC, -	     BoxPtr	    pbox, -	     int	    nbox, -	     int	    dx, -	     int	    dy, -	     Bool	    reverse, -	     Bool	    upsidedown) -{ -    ExaScreenPriv (pDstDrawable->pScreen); -    PixmapPtr pSrcPixmap, pDstPixmap; -    ExaPixmapPrivPtr pSrcExaPixmap, pDstExaPixmap; -    int	    src_off_x, src_off_y; -    int	    dst_off_x, dst_off_y; -    RegionPtr srcregion = NULL, dstregion = NULL; -    xRectangle *rects; -    Bool ret = TRUE; - -    /* avoid doing copy operations if no boxes */ -    if (nbox == 0) -	return TRUE; - -    pSrcPixmap = exaGetDrawablePixmap (pSrcDrawable); -    pDstPixmap = exaGetDrawablePixmap (pDstDrawable); - -    exaGetDrawableDeltas (pSrcDrawable, pSrcPixmap, &src_off_x, &src_off_y); -    exaGetDrawableDeltas (pDstDrawable, pDstPixmap, &dst_off_x, &dst_off_y); - -    rects = xalloc(nbox * sizeof(xRectangle)); - -    if (rects) { -	int i; -	int ordering; - -	for (i = 0; i < nbox; i++) { -	    rects[i].x = pbox[i].x1 + dx + src_off_x; -	    rects[i].y = pbox[i].y1 + dy + src_off_y; -	    rects[i].width = pbox[i].x2 - pbox[i].x1; -	    rects[i].height = pbox[i].y2 - pbox[i].y1; -	} - -	/* This must match the miRegionCopy() logic for reversing rect order */ -	if (nbox == 1 || (dx > 0 && dy > 0) || -	    (pDstDrawable != pSrcDrawable && -	     (pDstDrawable->type != DRAWABLE_WINDOW || -	      pSrcDrawable->type != DRAWABLE_WINDOW))) -	    ordering = CT_YXBANDED; -	else -	    ordering = CT_UNSORTED; - -	srcregion  = RECTS_TO_REGION(pScreen, nbox, rects, ordering); -	xfree(rects); - -	if (!pGC || !exaGCReadsDestination(pDstDrawable, pGC->planemask, -					   pGC->fillStyle, pGC->alu, -					   pGC->clientClipType)) { -	    dstregion = REGION_CREATE(pScreen, NullBox, 0); -	    REGION_COPY(pScreen, dstregion, srcregion); -	    REGION_TRANSLATE(pScreen, dstregion, dst_off_x - dx - src_off_x, -			     dst_off_y - dy - src_off_y); -	} -    } - - -    pSrcExaPixmap = ExaGetPixmapPriv (pSrcPixmap); -    pDstExaPixmap = ExaGetPixmapPriv (pDstPixmap); - -    /* Check whether the accelerator can use this pixmap. -     * If the pitch of the pixmaps is out of range, there's nothing -     * we can do but fall back to software rendering. -     */ -    if (pSrcExaPixmap->accel_blocked & EXA_RANGE_PITCH || -        pDstExaPixmap->accel_blocked & EXA_RANGE_PITCH) -	goto fallback; - -    /* If the width or the height of either of the pixmaps -     * is out of range, check whether the boxes are actually out of the -     * addressable range as well. If they aren't, we can still do -     * the copying in hardware. -     */ -    if (pSrcExaPixmap->accel_blocked || pDstExaPixmap->accel_blocked) { -        int i; - -        for (i = 0; i < nbox; i++) { -            /* src */ -            if ((pbox[i].x2 + dx + src_off_x) >= pExaScr->info->maxX || -                (pbox[i].y2 + dy + src_off_y) >= pExaScr->info->maxY) -                goto fallback; - -            /* dst */ -            if ((pbox[i].x2 + dst_off_x) >= pExaScr->info->maxX || -                (pbox[i].y2 + dst_off_y) >= pExaScr->info->maxY) -                goto fallback; -        } -    } - -    if (pExaScr->do_migration) { -	ExaMigrationRec pixmaps[2]; - -	pixmaps[0].as_dst = TRUE; -	pixmaps[0].as_src = FALSE; -	pixmaps[0].pPix = pDstPixmap; -	pixmaps[0].pReg = dstregion; -	pixmaps[1].as_dst = FALSE; -	pixmaps[1].as_src = TRUE; -	pixmaps[1].pPix = pSrcPixmap; -	pixmaps[1].pReg = srcregion; - -	exaDoMigration (pixmaps, 2, TRUE); -    } - -    /* Mixed directions must be handled specially if the card is lame */ -    if ((pExaScr->info->flags & EXA_TWO_BITBLT_DIRECTIONS) && -	reverse != upsidedown) { -	if (exaCopyNtoNTwoDir(pSrcDrawable, pDstDrawable, pGC, pbox, nbox, -			       dx, dy)) -	    goto out; -	goto fallback; -    } - -    if (exaPixmapHasGpuCopy(pDstPixmap)) { -	/* Normal blitting. */ -	if (exaPixmapHasGpuCopy(pSrcPixmap)) { -	    if (!(*pExaScr->info->PrepareCopy) (pSrcPixmap, pDstPixmap, reverse ? -1 : 1, -						upsidedown ? -1 : 1, -						pGC ? pGC->alu : GXcopy, -						pGC ? pGC->planemask : FB_ALLONES)) { -		goto fallback; -	    } - -	    while (nbox--) -	    { -		(*pExaScr->info->Copy) (pDstPixmap, -					pbox->x1 + dx + src_off_x, -					pbox->y1 + dy + src_off_y, -					pbox->x1 + dst_off_x, pbox->y1 + dst_off_y, -					pbox->x2 - pbox->x1, pbox->y2 - pbox->y1); -		pbox++; -	    } - -	    (*pExaScr->info->DoneCopy) (pDstPixmap); -	    exaMarkSync (pDstDrawable->pScreen); -	/* UTS: mainly for SHM PutImage's secondary path. -	 * -	 * Only taking this path for directly accessible pixmaps. -	 */ -	} else if (!pDstExaPixmap->pDamage) { -	    int bpp = pSrcDrawable->bitsPerPixel; -	    int src_stride = exaGetPixmapPitch(pSrcPixmap); -	    CARD8 *src = NULL; - -	    if (!pExaScr->info->UploadToScreen) -		goto fallback; - -	    if (pSrcDrawable->bitsPerPixel != pDstDrawable->bitsPerPixel) -		goto fallback; - -	    if (pSrcDrawable->bitsPerPixel < 8) -		goto fallback; - -	    if (pGC && !(pGC->alu == GXcopy && EXA_PM_IS_SOLID(pSrcDrawable,  pGC->planemask))) -		goto fallback; - -	    while (nbox--) -	    { -		src = pSrcExaPixmap->sys_ptr + (pbox->y1 + dy + src_off_y) * src_stride + (pbox->x1 + dx + src_off_x) * (bpp / 8); -		if (!pExaScr->info->UploadToScreen(pDstPixmap, pbox->x1 + dst_off_x, -				pbox->y1 + dst_off_y, pbox->x2 - pbox->x1, pbox->y2 - pbox->y1, -				(char *) src, src_stride)) -		    goto fallback; - -		pbox++; -	    } -	} else -	    goto fallback; -    } else -	goto fallback; - -    goto out; - -fallback: -    ret = FALSE; - -out: -    if (dstregion) { -	REGION_UNINIT(pScreen, dstregion); -	REGION_DESTROY(pScreen, dstregion); -    } -    if (srcregion) { -	REGION_UNINIT(pScreen, srcregion); -	REGION_DESTROY(pScreen, srcregion); -    } - -    return ret; -} - -void -exaCopyNtoN (DrawablePtr    pSrcDrawable, -	     DrawablePtr    pDstDrawable, -	     GCPtr	    pGC, -	     BoxPtr	    pbox, -	     int	    nbox, -	     int	    dx, -	     int	    dy, -	     Bool	    reverse, -	     Bool	    upsidedown, -	     Pixel	    bitplane, -	     void	    *closure) -{ -    ExaScreenPriv(pDstDrawable->pScreen); - -    if (pExaScr->fallback_counter || -	    (pExaScr->fallback_flags & EXA_FALLBACK_COPYWINDOW)) -	return; - -    if (exaHWCopyNtoN(pSrcDrawable, pDstDrawable, pGC, pbox, nbox, dx, dy, reverse, upsidedown)) -	return; - -    /* This is a CopyWindow, it's cleaner to fallback at the original call. */ -    if (pExaScr->fallback_flags & EXA_ACCEL_COPYWINDOW) { -	pExaScr->fallback_flags |= EXA_FALLBACK_COPYWINDOW; -	return; -    } - -    /* fallback */ -    ExaCheckCopyNtoN(pSrcDrawable, pDstDrawable, pGC, pbox, nbox, dx, dy, reverse, upsidedown, bitplane, closure); -} - -RegionPtr -exaCopyArea(DrawablePtr pSrcDrawable, DrawablePtr pDstDrawable, GCPtr pGC, -	    int srcx, int srcy, int width, int height, int dstx, int dsty) -{ -    ExaScreenPriv (pDstDrawable->pScreen); - -    if (pExaScr->fallback_counter || pExaScr->swappedOut) { -        return  ExaCheckCopyArea(pSrcDrawable, pDstDrawable, pGC, -                                 srcx, srcy, width, height, dstx, dsty); -    } - -    return  miDoCopy (pSrcDrawable, pDstDrawable, pGC, -                      srcx, srcy, width, height, -                      dstx, dsty, exaCopyNtoN, 0, NULL); -} - -static void -exaPolyPoint(DrawablePtr pDrawable, GCPtr pGC, int mode, int npt, -	     DDXPointPtr ppt) -{ -    ExaScreenPriv (pDrawable->pScreen); -    int i; -    xRectangle *prect; - -    /* If we can't reuse the current GC as is, don't bother accelerating the -     * points. -     */ -    if (pExaScr->fallback_counter || pGC->fillStyle != FillSolid) { -	ExaCheckPolyPoint(pDrawable, pGC, mode, npt, ppt); -	return; -    } - -    prect = xalloc(sizeof(xRectangle) * npt); -    for (i = 0; i < npt; i++) { -	prect[i].x = ppt[i].x; -	prect[i].y = ppt[i].y; -	if (i > 0 && mode == CoordModePrevious) { -	    prect[i].x += prect[i - 1].x; -	    prect[i].y += prect[i - 1].y; -	} -	prect[i].width = 1; -	prect[i].height = 1; -    } -    pGC->ops->PolyFillRect(pDrawable, pGC, npt, prect); -    xfree(prect); -} - -/** - * exaPolylines() checks if it can accelerate the lines as a group of - * horizontal or vertical lines (rectangles), and uses existing rectangle fill - * acceleration if so. - */ -static void -exaPolylines(DrawablePtr pDrawable, GCPtr pGC, int mode, int npt, -	     DDXPointPtr ppt) -{ -    ExaScreenPriv (pDrawable->pScreen); -    xRectangle *prect; -    int x1, x2, y1, y2; -    int i; - -    if (pExaScr->fallback_counter) { -	ExaCheckPolylines(pDrawable, pGC, mode, npt, ppt); -	return; -    } - -    /* Don't try to do wide lines or non-solid fill style. */ -    if (pGC->lineWidth != 0 || pGC->lineStyle != LineSolid || -	pGC->fillStyle != FillSolid) { -	ExaCheckPolylines(pDrawable, pGC, mode, npt, ppt); -	return; -    } - -    prect = xalloc(sizeof(xRectangle) * (npt - 1)); -    x1 = ppt[0].x; -    y1 = ppt[0].y; -    /* If we have any non-horizontal/vertical, fall back. */ -    for (i = 0; i < npt - 1; i++) { -	if (mode == CoordModePrevious) { -	    x2 = x1 + ppt[i + 1].x; -	    y2 = y1 + ppt[i + 1].y; -	} else { -	    x2 = ppt[i + 1].x; -	    y2 = ppt[i + 1].y; -	} - -	if (x1 != x2 && y1 != y2) { -	    xfree(prect); -	    ExaCheckPolylines(pDrawable, pGC, mode, npt, ppt); -	    return; -	} - -	if (x1 < x2) { -	    prect[i].x = x1; -	    prect[i].width = x2 - x1 + 1; -	} else { -	    prect[i].x = x2; -	    prect[i].width = x1 - x2 + 1; -	} -	if (y1 < y2) { -	    prect[i].y = y1; -	    prect[i].height = y2 - y1 + 1; -	} else { -	    prect[i].y = y2; -	    prect[i].height = y1 - y2 + 1; -	} - -	x1 = x2; -	y1 = y2; -    } -    pGC->ops->PolyFillRect(pDrawable, pGC, npt - 1, prect); -    xfree(prect); -} - -/** - * exaPolySegment() checks if it can accelerate the lines as a group of - * horizontal or vertical lines (rectangles), and uses existing rectangle fill - * acceleration if so. - */ -static void -exaPolySegment (DrawablePtr pDrawable, GCPtr pGC, int nseg, -		xSegment *pSeg) -{ -    ExaScreenPriv (pDrawable->pScreen); -    xRectangle *prect; -    int i; - -    /* Don't try to do wide lines or non-solid fill style. */ -    if (pExaScr->fallback_counter || pGC->lineWidth != 0 || -	pGC->lineStyle != LineSolid || pGC->fillStyle != FillSolid) -    { -	ExaCheckPolySegment(pDrawable, pGC, nseg, pSeg); -	return; -    } - -    /* If we have any non-horizontal/vertical, fall back. */ -    for (i = 0; i < nseg; i++) { -	if (pSeg[i].x1 != pSeg[i].x2 && pSeg[i].y1 != pSeg[i].y2) { -	    ExaCheckPolySegment(pDrawable, pGC, nseg, pSeg); -	    return; -	} -    } - -    prect = xalloc(sizeof(xRectangle) * nseg); -    for (i = 0; i < nseg; i++) { -	if (pSeg[i].x1 < pSeg[i].x2) { -	    prect[i].x = pSeg[i].x1; -	    prect[i].width = pSeg[i].x2 - pSeg[i].x1 + 1; -	} else { -	    prect[i].x = pSeg[i].x2; -	    prect[i].width = pSeg[i].x1 - pSeg[i].x2 + 1; -	} -	if (pSeg[i].y1 < pSeg[i].y2) { -	    prect[i].y = pSeg[i].y1; -	    prect[i].height = pSeg[i].y2 - pSeg[i].y1 + 1; -	} else { -	    prect[i].y = pSeg[i].y2; -	    prect[i].height = pSeg[i].y1 - pSeg[i].y2 + 1; -	} - -	/* don't paint last pixel */ -	if (pGC->capStyle == CapNotLast) { -	    if (prect[i].width == 1) -		prect[i].height--; -	    else -		prect[i].width--; -	} -    } -    pGC->ops->PolyFillRect(pDrawable, pGC, nseg, prect); -    xfree(prect); -} - -static Bool exaFillRegionSolid (DrawablePtr pDrawable, RegionPtr pRegion, -				Pixel pixel, CARD32 planemask, CARD32 alu, -				unsigned int clientClipType); - -static void -exaPolyFillRect(DrawablePtr pDrawable, -		GCPtr	    pGC, -		int	    nrect, -		xRectangle  *prect) -{ -    ExaScreenPriv (pDrawable->pScreen); -    RegionPtr	    pClip = fbGetCompositeClip(pGC); -    PixmapPtr	    pPixmap = exaGetDrawablePixmap(pDrawable); -    ExaPixmapPriv (pPixmap); -    register BoxPtr pbox; -    BoxPtr	    pextent; -    int		    extentX1, extentX2, extentY1, extentY2; -    int		    fullX1, fullX2, fullY1, fullY2; -    int		    partX1, partX2, partY1, partY2; -    int		    xoff, yoff; -    int		    xorg, yorg; -    int		    n; -    RegionPtr pReg = RECTS_TO_REGION(pScreen, nrect, prect, CT_UNSORTED); - -    /* Compute intersection of rects and clip region */ -    REGION_TRANSLATE(pScreen, pReg, pDrawable->x, pDrawable->y); -    REGION_INTERSECT(pScreen, pReg, pClip, pReg); - -    if (!REGION_NUM_RECTS(pReg)) { -	goto out; -    } - -    exaGetDrawableDeltas(pDrawable, pPixmap, &xoff, &yoff); - -    if (pExaScr->fallback_counter || pExaScr->swappedOut || -	    pExaPixmap->accel_blocked) -    { -	goto fallback; -    } - -    /* For ROPs where overlaps don't matter, convert rectangles to region and -     * call exaFillRegion{Solid,Tiled}. -     */ -    if ((pGC->fillStyle == FillSolid || pGC->fillStyle == FillTiled) && -	(nrect == 1 || pGC->alu == GXcopy || pGC->alu == GXclear || -	 pGC->alu == GXnoop || pGC->alu == GXcopyInverted || -	 pGC->alu == GXset)) { -	if (((pGC->fillStyle == FillSolid || pGC->tileIsPixel) && -	     exaFillRegionSolid(pDrawable, pReg, pGC->fillStyle == FillSolid ? -				pGC->fgPixel : pGC->tile.pixel,	pGC->planemask, -				pGC->alu, pGC->clientClipType)) || -	    (pGC->fillStyle == FillTiled && !pGC->tileIsPixel && -	     exaFillRegionTiled(pDrawable, pReg, pGC->tile.pixmap, &pGC->patOrg, -				pGC->planemask, pGC->alu, -				pGC->clientClipType))) { -	    goto out; -	} -    } - -    if (pGC->fillStyle != FillSolid && -	!(pGC->tileIsPixel && pGC->fillStyle == FillTiled)) -    { -	goto fallback; -    } - -    if (pExaScr->do_migration) { -	ExaMigrationRec pixmaps[1]; - -	pixmaps[0].as_dst = TRUE; -	pixmaps[0].as_src = FALSE; -	pixmaps[0].pPix = pPixmap; -	pixmaps[0].pReg = NULL; - -	exaDoMigration (pixmaps, 1, TRUE); -    } - -    if (!exaPixmapHasGpuCopy (pPixmap) || -	!(*pExaScr->info->PrepareSolid) (pPixmap, -					 pGC->alu, -					 pGC->planemask, -					 pGC->fgPixel)) -    { -fallback: -	ExaCheckPolyFillRect (pDrawable, pGC, nrect, prect); -	goto out; -    } - -    xorg = pDrawable->x; -    yorg = pDrawable->y; - -    pextent = REGION_EXTENTS(pGC->pScreen, pClip); -    extentX1 = pextent->x1; -    extentY1 = pextent->y1; -    extentX2 = pextent->x2; -    extentY2 = pextent->y2; -    while (nrect--) -    { -	fullX1 = prect->x + xorg; -	fullY1 = prect->y + yorg; -	fullX2 = fullX1 + (int) prect->width; -	fullY2 = fullY1 + (int) prect->height; -	prect++; - -	if (fullX1 < extentX1) -	    fullX1 = extentX1; - -	if (fullY1 < extentY1) -	    fullY1 = extentY1; - -	if (fullX2 > extentX2) -	    fullX2 = extentX2; - -	if (fullY2 > extentY2) -	    fullY2 = extentY2; - -	if ((fullX1 >= fullX2) || (fullY1 >= fullY2)) -	    continue; -	n = REGION_NUM_RECTS (pClip); -	if (n == 1) -	{ -	    (*pExaScr->info->Solid) (pPixmap, -				     fullX1 + xoff, fullY1 + yoff, -				     fullX2 + xoff, fullY2 + yoff); -	} -	else -	{ -	    pbox = REGION_RECTS(pClip); -	    /* -	     * clip the rectangle to each box in the clip region -	     * this is logically equivalent to calling Intersect(), -	     * but rectangles may overlap each other here. -	     */ -	    while(n--) -	    { -		partX1 = pbox->x1; -		if (partX1 < fullX1) -		    partX1 = fullX1; -		partY1 = pbox->y1; -		if (partY1 < fullY1) -		    partY1 = fullY1; -		partX2 = pbox->x2; -		if (partX2 > fullX2) -		    partX2 = fullX2; -		partY2 = pbox->y2; -		if (partY2 > fullY2) -		    partY2 = fullY2; - -		pbox++; - -		if (partX1 < partX2 && partY1 < partY2) { -		    (*pExaScr->info->Solid) (pPixmap, -					     partX1 + xoff, partY1 + yoff, -					     partX2 + xoff, partY2 + yoff); -		} -	    } -	} -    } -    (*pExaScr->info->DoneSolid) (pPixmap); -    exaMarkSync(pDrawable->pScreen); - -out: -    REGION_UNINIT(pScreen, pReg); -    REGION_DESTROY(pScreen, pReg); -} - -const GCOps exaOps = { -    exaFillSpans, -    ExaCheckSetSpans, -    exaPutImage, -    exaCopyArea, -    ExaCheckCopyPlane, -    exaPolyPoint, -    exaPolylines, -    exaPolySegment, -    miPolyRectangle, -    ExaCheckPolyArc, -    miFillPolygon, -    exaPolyFillRect, -    miPolyFillArc, -    miPolyText8, -    miPolyText16, -    miImageText8, -    miImageText16, -    ExaCheckImageGlyphBlt, -    ExaCheckPolyGlyphBlt, -    ExaCheckPushPixels, -}; - -void -exaCopyWindow(WindowPtr pWin, DDXPointRec ptOldOrg, RegionPtr prgnSrc) -{ -    RegionRec	rgnDst; -    int		dx, dy; -    PixmapPtr	pPixmap = (*pWin->drawable.pScreen->GetWindowPixmap) (pWin); -    ExaScreenPriv(pWin->drawable.pScreen); - -    dx = ptOldOrg.x - pWin->drawable.x; -    dy = ptOldOrg.y - pWin->drawable.y; -    REGION_TRANSLATE(pWin->drawable.pScreen, prgnSrc, -dx, -dy); - -    REGION_INIT (pWin->drawable.pScreen, &rgnDst, NullBox, 0); - -    REGION_INTERSECT(pWin->drawable.pScreen, &rgnDst, &pWin->borderClip, prgnSrc); -#ifdef COMPOSITE -    if (pPixmap->screen_x || pPixmap->screen_y) -	REGION_TRANSLATE (pWin->drawable.pScreen, &rgnDst, -			  -pPixmap->screen_x, -pPixmap->screen_y); -#endif - -    if (pExaScr->fallback_counter) { -	pExaScr->fallback_flags |= EXA_FALLBACK_COPYWINDOW; -	goto fallback; -    } - -    pExaScr->fallback_flags |= EXA_ACCEL_COPYWINDOW; -    miCopyRegion (&pPixmap->drawable, &pPixmap->drawable, -		  NULL, -		  &rgnDst, dx, dy, exaCopyNtoN, 0, NULL); -    pExaScr->fallback_flags &= ~EXA_ACCEL_COPYWINDOW; - -fallback: -    REGION_UNINIT(pWin->drawable.pScreen, &rgnDst); - -    if (pExaScr->fallback_flags & EXA_FALLBACK_COPYWINDOW) { -	pExaScr->fallback_flags &= ~EXA_FALLBACK_COPYWINDOW; -	REGION_TRANSLATE(pWin->drawable.pScreen, prgnSrc, dx, dy); -	ExaCheckCopyWindow(pWin, ptOldOrg, prgnSrc); -    } -} - -static Bool -exaFillRegionSolid (DrawablePtr	pDrawable, RegionPtr pRegion, Pixel pixel, -		    CARD32 planemask, CARD32 alu, unsigned int clientClipType) -{ -    ExaScreenPriv(pDrawable->pScreen); -    PixmapPtr pPixmap = exaGetDrawablePixmap (pDrawable); -    ExaPixmapPriv (pPixmap); -    int xoff, yoff; -    Bool ret = FALSE; - -    exaGetDrawableDeltas(pDrawable, pPixmap, &xoff, &yoff); -    REGION_TRANSLATE(pScreen, pRegion, xoff, yoff); - -    if (pExaScr->fallback_counter || pExaPixmap->accel_blocked) -	goto out; - -    if (pExaScr->do_migration) { -	ExaMigrationRec pixmaps[1]; - -	pixmaps[0].as_dst = TRUE; -	pixmaps[0].as_src = FALSE; -	pixmaps[0].pPix = pPixmap; -	pixmaps[0].pReg = exaGCReadsDestination(pDrawable, planemask, FillSolid, -						alu, clientClipType) ? NULL : pRegion; - -	exaDoMigration (pixmaps, 1, TRUE); -    } - -    if (exaPixmapHasGpuCopy (pPixmap) && -	(*pExaScr->info->PrepareSolid) (pPixmap, alu, planemask, pixel)) -    { -	int nbox; -	BoxPtr pBox; - -	nbox = REGION_NUM_RECTS (pRegion); -	pBox = REGION_RECTS (pRegion); - -	while (nbox--) -	{ -	    (*pExaScr->info->Solid) (pPixmap, pBox->x1, pBox->y1, pBox->x2, -				     pBox->y2); -	    pBox++; -	} -	(*pExaScr->info->DoneSolid) (pPixmap); -	exaMarkSync(pDrawable->pScreen); - -	if (pExaPixmap->pDamage && -	    pExaPixmap->sys_ptr && pDrawable->type == DRAWABLE_PIXMAP && -	    pDrawable->width == 1 && pDrawable->height == 1 && -	    pDrawable->bitsPerPixel != 24) { -	    ExaPixmapPriv(pPixmap); -	    RegionPtr pending_damage = DamagePendingRegion(pExaPixmap->pDamage); - -	    switch (pDrawable->bitsPerPixel) { -	    case 32: -		*(CARD32*)pExaPixmap->sys_ptr = pixel; -		break; -	    case 16: -		*(CARD16*)pExaPixmap->sys_ptr = pixel; -		break; -	    case 8: -		*(CARD8*)pExaPixmap->sys_ptr = pixel; -	    } - -	    REGION_UNION(pScreen, &pExaPixmap->validSys, &pExaPixmap->validSys, -			 pRegion); -	    REGION_UNION(pScreen, &pExaPixmap->validFB, &pExaPixmap->validFB, -			 pRegion); -	    REGION_SUBTRACT(pScreen, pending_damage, pending_damage, pRegion); -	} - -	ret = TRUE; -    } - -out: -    REGION_TRANSLATE(pScreen, pRegion, -xoff, -yoff); - -    return ret; -} - -/* Try to do an accelerated tile of the pTile into pRegion of pDrawable. - * Based on fbFillRegionTiled(), fbTile(). - */ -Bool -exaFillRegionTiled (DrawablePtr pDrawable, RegionPtr pRegion, PixmapPtr pTile, -		    DDXPointPtr pPatOrg, CARD32 planemask, CARD32 alu, -		    unsigned int clientClipType) -{ -    ExaScreenPriv(pDrawable->pScreen); -    PixmapPtr pPixmap; -    ExaPixmapPrivPtr pExaPixmap; -    ExaPixmapPrivPtr pTileExaPixmap = ExaGetPixmapPriv(pTile); -    int xoff, yoff; -    int tileWidth, tileHeight; -    int nbox = REGION_NUM_RECTS (pRegion); -    BoxPtr pBox = REGION_RECTS (pRegion); -    Bool ret = FALSE; -    int i; - -    tileWidth = pTile->drawable.width; -    tileHeight = pTile->drawable.height; - -    /* If we're filling with a solid color, grab it out and go to -     * FillRegionSolid, saving numerous copies. -     */ -    if (tileWidth == 1 && tileHeight == 1) -	return exaFillRegionSolid(pDrawable, pRegion, -				  exaGetPixmapFirstPixel (pTile), planemask, -				  alu, clientClipType); - -    pPixmap = exaGetDrawablePixmap (pDrawable); -    pExaPixmap = ExaGetPixmapPriv (pPixmap); - -    if (pExaScr->fallback_counter || pExaPixmap->accel_blocked || -	    pTileExaPixmap->accel_blocked) -	return FALSE; - -    if (pExaScr->do_migration) { -	ExaMigrationRec pixmaps[2]; - -	pixmaps[0].as_dst = TRUE; -	pixmaps[0].as_src = FALSE; -	pixmaps[0].pPix = pPixmap; -	pixmaps[0].pReg = exaGCReadsDestination(pDrawable, planemask, FillTiled, -						alu, clientClipType) ? NULL : pRegion; -	pixmaps[1].as_dst = FALSE; -	pixmaps[1].as_src = TRUE; -	pixmaps[1].pPix = pTile; -	pixmaps[1].pReg = NULL; - -	exaDoMigration (pixmaps, 2, TRUE); -    } - -    pPixmap = exaGetOffscreenPixmap (pDrawable, &xoff, &yoff); - -    if (!pPixmap || !exaPixmapHasGpuCopy(pTile)) -	return FALSE; - -    if ((*pExaScr->info->PrepareCopy) (pTile, pPixmap, 1, 1, alu, planemask)) -    { -	if (xoff || yoff) -	    REGION_TRANSLATE(pScreen, pRegion, xoff, yoff); - -	for (i = 0; i < nbox; i++) -	{ -	    int height = pBox[i].y2 - pBox[i].y1; -	    int dstY = pBox[i].y1; -	    int tileY; - -	    if (alu == GXcopy) -		height = min(height, tileHeight); - -	    modulus(dstY - yoff - pDrawable->y - pPatOrg->y, tileHeight, tileY); - -	    while (height > 0) { -		int width = pBox[i].x2 - pBox[i].x1; -		int dstX = pBox[i].x1; -		int tileX; -		int h = tileHeight - tileY; - -		if (alu == GXcopy) -		    width = min(width, tileWidth); - -		if (h > height) -		    h = height; -		height -= h; - -		modulus(dstX - xoff - pDrawable->x - pPatOrg->x, tileWidth, -			tileX); - -		while (width > 0) { -		    int w = tileWidth - tileX; -		    if (w > width) -			w = width; -		    width -= w; - -		    (*pExaScr->info->Copy) (pPixmap, tileX, tileY, dstX, dstY, -					    w, h); -		    dstX += w; -		    tileX = 0; -		} -		dstY += h; -		tileY = 0; -	    } -	} -	(*pExaScr->info->DoneCopy) (pPixmap); - -	/* With GXcopy, we only need to do the basic algorithm up to the tile -	 * size; then, we can just keep doubling the destination in each -	 * direction until it fills the box. This way, the number of copy -	 * operations is O(log(rx)) + O(log(ry)) instead of O(rx * ry), where -	 * rx/ry is the ratio between box and tile width/height. This can make -	 * a big difference if each driver copy incurs a significant constant -	 * overhead. -	 */ -	if (alu != GXcopy) -	    ret = TRUE; -	else { -	    Bool more_copy = FALSE; - -	    for (i = 0; i < nbox; i++) { -		int dstX = pBox[i].x1 + tileWidth; -		int dstY = pBox[i].y1 + tileHeight; - -		if ((dstX < pBox[i].x2) || (dstY < pBox[i].y2)) { -		    more_copy = TRUE; -		    break; -		} -	    } - -	    if (more_copy == FALSE) -		ret = TRUE; - -	    if (more_copy && (*pExaScr->info->PrepareCopy) (pPixmap, pPixmap, -							    1, 1, alu, planemask)) { -		for (i = 0; i < nbox; i++) -		{ -		    int dstX = pBox[i].x1 + tileWidth; -		    int dstY = pBox[i].y1 + tileHeight; -		    int width = min(pBox[i].x2 - dstX, tileWidth); -		    int height = min(pBox[i].y2 - pBox[i].y1, tileHeight); - -		    while (dstX < pBox[i].x2) { -			(*pExaScr->info->Copy) (pPixmap, pBox[i].x1, pBox[i].y1, -						dstX, pBox[i].y1, width, height); -			dstX += width; -			width = min(pBox[i].x2 - dstX, width * 2); -		    } - -		    width = pBox[i].x2 - pBox[i].x1; -		    height = min(pBox[i].y2 - dstY, tileHeight); - -		    while (dstY < pBox[i].y2) { -			(*pExaScr->info->Copy) (pPixmap, pBox[i].x1, pBox[i].y1, -						pBox[i].x1, dstY, width, height); -			dstY += height; -			height = min(pBox[i].y2 - dstY, height * 2); -		    } -		} - -		(*pExaScr->info->DoneCopy) (pPixmap); - -		ret = TRUE; -	    } -	} - -	exaMarkSync(pDrawable->pScreen); - -	if (xoff || yoff) -	    REGION_TRANSLATE(pScreen, pRegion, -xoff, -yoff); -    } - -    return ret; -} - - -/** - * Accelerates GetImage for solid ZPixmap downloads from framebuffer memory. - * - * This is probably the only case we actually care about.  The rest fall through - * to migration and fbGetImage, which hopefully will result in migration pushing - * the pixmap out of framebuffer. - */ -void -exaGetImage (DrawablePtr pDrawable, int x, int y, int w, int h, -	     unsigned int format, unsigned long planeMask, char *d) -{ -    ExaScreenPriv (pDrawable->pScreen); -    PixmapPtr pPix = exaGetDrawablePixmap (pDrawable); -    ExaPixmapPriv(pPix); -    int xoff, yoff; -    Bool ok; - -    if (pExaScr->fallback_counter || pExaScr->swappedOut) -	goto fallback; - -    /* If there's a system copy, we want to save the result there */ -    if (pExaPixmap->pDamage) -	goto fallback; - -    pPix = exaGetOffscreenPixmap (pDrawable, &xoff, &yoff); - -    if (pPix == NULL || pExaScr->info->DownloadFromScreen == NULL) -	goto fallback; - -    /* Only cover the ZPixmap, solid copy case. */ -    if (format != ZPixmap || !EXA_PM_IS_SOLID(pDrawable, planeMask)) -	goto fallback; - -    /* Only try to handle the 8bpp and up cases, since we don't want to think -     * about <8bpp. -     */ -    if (pDrawable->bitsPerPixel < 8) -	goto fallback; - -    ok = pExaScr->info->DownloadFromScreen(pPix, pDrawable->x + x + xoff, -					   pDrawable->y + y + yoff, w, h, d, -					   PixmapBytePad(w, pDrawable->depth)); -    if (ok) { -	exaWaitSync(pDrawable->pScreen); -	return; -    } - -fallback: -    ExaCheckGetImage(pDrawable, x, y, w, h, format, planeMask, d); -} +/*
 + * Copyright © 2001 Keith Packard
 + *
 + * Partly based on code that is Copyright © The XFree86 Project Inc.
 + *
 + * 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.
 + *
 + * Authors:
 + *    Eric Anholt <eric@anholt.net>
 + *    Michel Dänzer <michel@tungstengraphics.com>
 + *
 + */
 +
 +#ifdef HAVE_DIX_CONFIG_H
 +#include <dix-config.h>
 +#endif
 +#include "exa_priv.h"
 +#include <X11/fonts/fontstruct.h>
 +#include "dixfontstr.h"
 +#include "exa.h"
 +
 +static void
 +exaFillSpans(DrawablePtr pDrawable, GCPtr pGC, int n,
 +	     DDXPointPtr ppt, int *pwidth, int fSorted)
 +{
 +    ScreenPtr	    pScreen = pDrawable->pScreen;
 +    ExaScreenPriv (pScreen);
 +    RegionPtr	    pClip = fbGetCompositeClip(pGC);
 +    PixmapPtr	    pPixmap = exaGetDrawablePixmap (pDrawable);
 +    ExaPixmapPriv (pPixmap);
 +    BoxPtr	    pextent, pbox;
 +    int		    nbox;
 +    int		    extentX1, extentX2, extentY1, extentY2;
 +    int		    fullX1, fullX2, fullY1;
 +    int		    partX1, partX2;
 +    int		    off_x, off_y;
 +
 +    if (pExaScr->fallback_counter ||
 +	pExaScr->swappedOut ||
 +	pGC->fillStyle != FillSolid ||
 +	pExaPixmap->accel_blocked)
 +    {
 +	ExaCheckFillSpans (pDrawable, pGC, n, ppt, pwidth, fSorted);
 +	return;
 +    }
 +
 +    if (pExaScr->do_migration) {
 +	ExaMigrationRec pixmaps[1];
 +
 +	pixmaps[0].as_dst = TRUE;
 +	pixmaps[0].as_src = FALSE;
 +	pixmaps[0].pPix = pPixmap;
 +	pixmaps[0].pReg = NULL;
 +
 +	exaDoMigration (pixmaps, 1, TRUE);
 +    }
 +
 +    if (!(pPixmap = exaGetOffscreenPixmap (pDrawable, &off_x, &off_y)) ||
 +	!(*pExaScr->info->PrepareSolid) (pPixmap,
 +					 pGC->alu,
 +					 pGC->planemask,
 +					 pGC->fgPixel))
 +    {
 +	ExaCheckFillSpans (pDrawable, pGC, n, ppt, pwidth, fSorted);
 +	return;
 +    }
 +
 +    pextent = REGION_EXTENTS(pGC->pScreen, pClip);
 +    extentX1 = pextent->x1;
 +    extentY1 = pextent->y1;
 +    extentX2 = pextent->x2;
 +    extentY2 = pextent->y2;
 +    while (n--)
 +    {
 +	fullX1 = ppt->x;
 +	fullY1 = ppt->y;
 +	fullX2 = fullX1 + (int) *pwidth;
 +	ppt++;
 +	pwidth++;
 +
 +	if (fullY1 < extentY1 || extentY2 <= fullY1)
 +	    continue;
 +
 +	if (fullX1 < extentX1)
 +	    fullX1 = extentX1;
 +
 +	if (fullX2 > extentX2)
 +	    fullX2 = extentX2;
 +
 +	if (fullX1 >= fullX2)
 +	    continue;
 +
 +	nbox = REGION_NUM_RECTS (pClip);
 +	if (nbox == 1)
 +	{
 +	    (*pExaScr->info->Solid) (pPixmap,
 +				     fullX1 + off_x, fullY1 + off_y,
 +				     fullX2 + off_x, fullY1 + 1 + off_y);
 +	}
 +	else
 +	{
 +	    pbox = REGION_RECTS(pClip);
 +	    while(nbox--)
 +	    {
 +		if (pbox->y1 <= fullY1 && fullY1 < pbox->y2)
 +		{
 +		    partX1 = pbox->x1;
 +		    if (partX1 < fullX1)
 +			partX1 = fullX1;
 +		    partX2 = pbox->x2;
 +		    if (partX2 > fullX2)
 +			partX2 = fullX2;
 +		    if (partX2 > partX1) {
 +			(*pExaScr->info->Solid) (pPixmap,
 +						 partX1 + off_x, fullY1 + off_y,
 +						 partX2 + off_x, fullY1 + 1 + off_y);
 +		    }
 +		}
 +		pbox++;
 +	    }
 +	}
 +    }
 +    (*pExaScr->info->DoneSolid) (pPixmap);
 +    exaMarkSync(pScreen);
 +}
 +
 +static Bool
 +exaDoPutImage (DrawablePtr pDrawable, GCPtr pGC, int depth, int x, int y,
 +	       int w, int h, int format, char *bits, int src_stride)
 +{
 +    ExaScreenPriv (pDrawable->pScreen);
 +    PixmapPtr pPix = exaGetDrawablePixmap (pDrawable);
 +    ExaPixmapPriv(pPix);
 +    RegionPtr pClip;
 +    BoxPtr pbox;
 +    int nbox;
 +    int xoff, yoff;
 +    int bpp = pDrawable->bitsPerPixel;
 +    Bool ret = TRUE;
 +
 +    if (pExaScr->fallback_counter || pExaPixmap->accel_blocked || !pExaScr->info->UploadToScreen)
 +	return FALSE;
 +
 +    /* If there's a system copy, we want to save the result there */
 +    if (pExaPixmap->pDamage)
 +	return FALSE;
 +
 +    /* Don't bother with under 8bpp, XYPixmaps. */
 +    if (format != ZPixmap || bpp < 8)
 +	return FALSE;
 +
 +    /* Only accelerate copies: no rop or planemask. */
 +    if (!EXA_PM_IS_SOLID(pDrawable, pGC->planemask) || pGC->alu != GXcopy)
 +	return FALSE;
 +
 +    if (pExaScr->swappedOut)
 +	return FALSE;
 +
 +    if (pExaScr->do_migration) {
 +	ExaMigrationRec pixmaps[1];
 +
 +	pixmaps[0].as_dst = TRUE;
 +	pixmaps[0].as_src = FALSE;
 +	pixmaps[0].pPix = pPix;
 +	pixmaps[0].pReg = DamagePendingRegion(pExaPixmap->pDamage);
 +
 +	exaDoMigration (pixmaps, 1, TRUE);
 +    }
 +
 +    pPix = exaGetOffscreenPixmap (pDrawable, &xoff, &yoff);
 +
 +    if (!pPix)
 +	return FALSE;
 +
 +    x += pDrawable->x;
 +    y += pDrawable->y;
 +
 +    pClip = fbGetCompositeClip(pGC);
 +    for (nbox = REGION_NUM_RECTS(pClip),
 +	 pbox = REGION_RECTS(pClip);
 +	 nbox--;
 +	 pbox++)
 +    {
 +	int x1 = x;
 +	int y1 = y;
 +	int x2 = x + w;
 +	int y2 = y + h;
 +	char *src;
 +	Bool ok;
 +
 +	if (x1 < pbox->x1)
 +	    x1 = pbox->x1;
 +	if (y1 < pbox->y1)
 +	    y1 = pbox->y1;
 +	if (x2 > pbox->x2)
 +	    x2 = pbox->x2;
 +	if (y2 > pbox->y2)
 +	    y2 = pbox->y2;
 +	if (x1 >= x2 || y1 >= y2)
 +	    continue;
 +
 +	src = bits + (y1 - y) * src_stride + (x1 - x) * (bpp / 8);
 +	ok = pExaScr->info->UploadToScreen(pPix, x1 + xoff, y1 + yoff,
 +					   x2 - x1, y2 - y1, src, src_stride);
 +	/* We have to fall back completely, and ignore what has already been completed.
 +	 * Messing with the fb layer directly like we used to is completely unacceptable.
 +	 */
 +	if (!ok) {
 +	    ret = FALSE;
 +	    break;
 +	}
 +    }
 +
 +    if (ret)
 +	exaMarkSync(pDrawable->pScreen);
 +
 +    return ret;
 +}
 +
 +static void
 +exaPutImage (DrawablePtr pDrawable, GCPtr pGC, int depth, int x, int y,
 +	     int w, int h, int leftPad, int format, char *bits)
 +{
 +    if (!exaDoPutImage(pDrawable, pGC, depth, x, y, w, h, format, bits,
 +		       PixmapBytePad(w, pDrawable->depth)))
 +	ExaCheckPutImage(pDrawable, pGC, depth, x, y, w, h, leftPad, format,
 +			 bits);
 +}
 +
 +static Bool __inline
 +exaCopyNtoNTwoDir (DrawablePtr pSrcDrawable, DrawablePtr pDstDrawable,
 +		   GCPtr pGC, BoxPtr pbox, int nbox, int dx, int dy)
 +{
 +    ExaScreenPriv (pDstDrawable->pScreen);
 +    PixmapPtr pSrcPixmap, pDstPixmap;
 +    int src_off_x, src_off_y, dst_off_x, dst_off_y;
 +    int dirsetup;
 +
 +    /* Need to get both pixmaps to call the driver routines */
 +    pSrcPixmap = exaGetOffscreenPixmap (pSrcDrawable, &src_off_x, &src_off_y);
 +    pDstPixmap = exaGetOffscreenPixmap (pDstDrawable, &dst_off_x, &dst_off_y);
 +    if (!pSrcPixmap || !pDstPixmap)
 +	return FALSE;
 +
 +    /*
 +     * Now the case of a chip that only supports xdir = ydir = 1 or
 +     * xdir = ydir = -1, but we have xdir != ydir.
 +     */
 +    dirsetup = 0;	/* No direction set up yet. */
 +    for (; nbox; pbox++, nbox--) {
 +	if (dx >= 0 && (src_off_y + pbox->y1 + dy) != pbox->y1) {
 +	    /* Do a xdir = ydir = -1 blit instead. */
 +	    if (dirsetup != -1) {
 +		if (dirsetup != 0)
 +		    pExaScr->info->DoneCopy(pDstPixmap);
 +		dirsetup = -1;
 +		if (!(*pExaScr->info->PrepareCopy)(pSrcPixmap,
 +						   pDstPixmap,
 +						   -1, -1,
 +						   pGC ? pGC->alu : GXcopy,
 +						   pGC ? pGC->planemask :
 +							 FB_ALLONES))
 +		    return FALSE;
 +	    }
 +	    (*pExaScr->info->Copy)(pDstPixmap,
 +				   src_off_x + pbox->x1 + dx,
 +				   src_off_y + pbox->y1 + dy,
 +				   dst_off_x + pbox->x1,
 +				   dst_off_y + pbox->y1,
 +				   pbox->x2 - pbox->x1,
 +				   pbox->y2 - pbox->y1);
 +	} else if (dx < 0 && (src_off_y + pbox->y1 + dy) != pbox->y1) {
 +	    /* Do a xdir = ydir = 1 blit instead. */
 +	    if (dirsetup != 1) {
 +		if (dirsetup != 0)
 +		    pExaScr->info->DoneCopy(pDstPixmap);
 +		dirsetup = 1;
 +		if (!(*pExaScr->info->PrepareCopy)(pSrcPixmap,
 +						   pDstPixmap,
 +						   1, 1,
 +						   pGC ? pGC->alu : GXcopy,
 +						   pGC ? pGC->planemask :
 +							 FB_ALLONES))
 +		    return FALSE;
 +	    }
 +	    (*pExaScr->info->Copy)(pDstPixmap,
 +				   src_off_x + pbox->x1 + dx,
 +				   src_off_y + pbox->y1 + dy,
 +				   dst_off_x + pbox->x1,
 +				   dst_off_y + pbox->y1,
 +				   pbox->x2 - pbox->x1,
 +				   pbox->y2 - pbox->y1);
 +	} else if (dx >= 0) {
 +	    /*
 +	     * xdir = 1, ydir = -1.
 +	     * Perform line-by-line xdir = ydir = 1 blits, going up.
 +	     */
 +	    int i;
 +	    if (dirsetup != 1) {
 +		if (dirsetup != 0)
 +		    pExaScr->info->DoneCopy(pDstPixmap);
 +		dirsetup = 1;
 +		if (!(*pExaScr->info->PrepareCopy)(pSrcPixmap,
 +						   pDstPixmap,
 +						   1, 1,
 +						   pGC ? pGC->alu : GXcopy,
 +						   pGC ? pGC->planemask :
 +							 FB_ALLONES))
 +		    return FALSE;
 +	    }
 +	    for (i = pbox->y2 - pbox->y1 - 1; i >= 0; i--)
 +		(*pExaScr->info->Copy)(pDstPixmap,
 +				       src_off_x + pbox->x1 + dx,
 +				       src_off_y + pbox->y1 + dy + i,
 +				       dst_off_x + pbox->x1,
 +				       dst_off_y + pbox->y1 + i,
 +				       pbox->x2 - pbox->x1, 1);
 +	} else {
 +	    /*
 +	     * xdir = -1, ydir = 1.
 +	     * Perform line-by-line xdir = ydir = -1 blits, going down.
 +	     */
 +	    int i;
 +	    if (dirsetup != -1) {
 +		if (dirsetup != 0)
 +		    pExaScr->info->DoneCopy(pDstPixmap);
 +		dirsetup = -1;
 +		if (!(*pExaScr->info->PrepareCopy)(pSrcPixmap,
 +						   pDstPixmap,
 +						   -1, -1,
 +						   pGC ? pGC->alu : GXcopy,
 +						   pGC ? pGC->planemask :
 +							 FB_ALLONES))
 +		    return FALSE;
 +	    }
 +	    for (i = 0; i < pbox->y2 - pbox->y1; i++)
 +		(*pExaScr->info->Copy)(pDstPixmap,
 +				       src_off_x + pbox->x1 + dx,
 +				       src_off_y + pbox->y1 + dy + i,
 +				       dst_off_x + pbox->x1,
 +				       dst_off_y + pbox->y1 + i,
 +				       pbox->x2 - pbox->x1, 1);
 +	}
 +    }
 +    if (dirsetup != 0)
 +	pExaScr->info->DoneCopy(pDstPixmap);
 +    exaMarkSync(pDstDrawable->pScreen);
 +    return TRUE;
 +}
 +
 +Bool
 +exaHWCopyNtoN (DrawablePtr    pSrcDrawable,
 +	     DrawablePtr    pDstDrawable,
 +	     GCPtr	    pGC,
 +	     BoxPtr	    pbox,
 +	     int	    nbox,
 +	     int	    dx,
 +	     int	    dy,
 +	     Bool	    reverse,
 +	     Bool	    upsidedown)
 +{
 +    ExaScreenPriv (pDstDrawable->pScreen);
 +    PixmapPtr pSrcPixmap, pDstPixmap;
 +    ExaPixmapPrivPtr pSrcExaPixmap, pDstExaPixmap;
 +    int	    src_off_x, src_off_y;
 +    int	    dst_off_x, dst_off_y;
 +    RegionPtr srcregion = NULL, dstregion = NULL;
 +    xRectangle *rects;
 +    Bool ret = TRUE;
 +
 +    /* avoid doing copy operations if no boxes */
 +    if (nbox == 0)
 +	return TRUE;
 +
 +    pSrcPixmap = exaGetDrawablePixmap (pSrcDrawable);
 +    pDstPixmap = exaGetDrawablePixmap (pDstDrawable);
 +
 +    exaGetDrawableDeltas (pSrcDrawable, pSrcPixmap, &src_off_x, &src_off_y);
 +    exaGetDrawableDeltas (pDstDrawable, pDstPixmap, &dst_off_x, &dst_off_y);
 +
 +    rects = xalloc(nbox * sizeof(xRectangle));
 +
 +    if (rects) {
 +	int i;
 +	int ordering;
 +
 +	for (i = 0; i < nbox; i++) {
 +	    rects[i].x = pbox[i].x1 + dx + src_off_x;
 +	    rects[i].y = pbox[i].y1 + dy + src_off_y;
 +	    rects[i].width = pbox[i].x2 - pbox[i].x1;
 +	    rects[i].height = pbox[i].y2 - pbox[i].y1;
 +	}
 +
 +	/* This must match the miRegionCopy() logic for reversing rect order */
 +	if (nbox == 1 || (dx > 0 && dy > 0) ||
 +	    (pDstDrawable != pSrcDrawable &&
 +	     (pDstDrawable->type != DRAWABLE_WINDOW ||
 +	      pSrcDrawable->type != DRAWABLE_WINDOW)))
 +	    ordering = CT_YXBANDED;
 +	else
 +	    ordering = CT_UNSORTED;
 +
 +	srcregion  = RECTS_TO_REGION(pScreen, nbox, rects, ordering);
 +	xfree(rects);
 +
 +	if (!pGC || !exaGCReadsDestination(pDstDrawable, pGC->planemask,
 +					   pGC->fillStyle, pGC->alu,
 +					   pGC->clientClipType)) {
 +	    dstregion = REGION_CREATE(pScreen, NullBox, 0);
 +	    REGION_COPY(pScreen, dstregion, srcregion);
 +	    REGION_TRANSLATE(pScreen, dstregion, dst_off_x - dx - src_off_x,
 +			     dst_off_y - dy - src_off_y);
 +	}
 +    }
 +
 +
 +    pSrcExaPixmap = ExaGetPixmapPriv (pSrcPixmap);
 +    pDstExaPixmap = ExaGetPixmapPriv (pDstPixmap);
 +
 +    /* Check whether the accelerator can use this pixmap.
 +     * If the pitch of the pixmaps is out of range, there's nothing
 +     * we can do but fall back to software rendering.
 +     */
 +    if (pSrcExaPixmap->accel_blocked & EXA_RANGE_PITCH ||
 +        pDstExaPixmap->accel_blocked & EXA_RANGE_PITCH)
 +	goto fallback;
 +
 +    /* If the width or the height of either of the pixmaps
 +     * is out of range, check whether the boxes are actually out of the
 +     * addressable range as well. If they aren't, we can still do
 +     * the copying in hardware.
 +     */
 +    if (pSrcExaPixmap->accel_blocked || pDstExaPixmap->accel_blocked) {
 +        int i;
 +
 +        for (i = 0; i < nbox; i++) {
 +            /* src */
 +            if ((pbox[i].x2 + dx + src_off_x) >= pExaScr->info->maxX ||
 +                (pbox[i].y2 + dy + src_off_y) >= pExaScr->info->maxY)
 +                goto fallback;
 +
 +            /* dst */
 +            if ((pbox[i].x2 + dst_off_x) >= pExaScr->info->maxX ||
 +                (pbox[i].y2 + dst_off_y) >= pExaScr->info->maxY)
 +                goto fallback;
 +        }
 +    }
 +
 +    if (pExaScr->do_migration) {
 +	ExaMigrationRec pixmaps[2];
 +
 +	pixmaps[0].as_dst = TRUE;
 +	pixmaps[0].as_src = FALSE;
 +	pixmaps[0].pPix = pDstPixmap;
 +	pixmaps[0].pReg = dstregion;
 +	pixmaps[1].as_dst = FALSE;
 +	pixmaps[1].as_src = TRUE;
 +	pixmaps[1].pPix = pSrcPixmap;
 +	pixmaps[1].pReg = srcregion;
 +
 +	exaDoMigration (pixmaps, 2, TRUE);
 +    }
 +
 +    /* Mixed directions must be handled specially if the card is lame */
 +    if ((pExaScr->info->flags & EXA_TWO_BITBLT_DIRECTIONS) &&
 +	reverse != upsidedown) {
 +	if (exaCopyNtoNTwoDir(pSrcDrawable, pDstDrawable, pGC, pbox, nbox,
 +			       dx, dy))
 +	    goto out;
 +	goto fallback;
 +    }
 +
 +    if (exaPixmapHasGpuCopy(pDstPixmap)) {
 +	/* Normal blitting. */
 +	if (exaPixmapHasGpuCopy(pSrcPixmap)) {
 +	    if (!(*pExaScr->info->PrepareCopy) (pSrcPixmap, pDstPixmap, reverse ? -1 : 1,
 +						upsidedown ? -1 : 1,
 +						pGC ? pGC->alu : GXcopy,
 +						pGC ? pGC->planemask : FB_ALLONES)) {
 +		goto fallback;
 +	    }
 +
 +	    while (nbox--)
 +	    {
 +		(*pExaScr->info->Copy) (pDstPixmap,
 +					pbox->x1 + dx + src_off_x,
 +					pbox->y1 + dy + src_off_y,
 +					pbox->x1 + dst_off_x, pbox->y1 + dst_off_y,
 +					pbox->x2 - pbox->x1, pbox->y2 - pbox->y1);
 +		pbox++;
 +	    }
 +
 +	    (*pExaScr->info->DoneCopy) (pDstPixmap);
 +	    exaMarkSync (pDstDrawable->pScreen);
 +	/* UTS: mainly for SHM PutImage's secondary path.
 +	 *
 +	 * Only taking this path for directly accessible pixmaps.
 +	 */
 +	} else if (!pDstExaPixmap->pDamage) {
 +	    int bpp = pSrcDrawable->bitsPerPixel;
 +	    int src_stride = exaGetPixmapPitch(pSrcPixmap);
 +	    CARD8 *src = NULL;
 +
 +	    if (!pExaScr->info->UploadToScreen)
 +		goto fallback;
 +
 +	    if (pSrcDrawable->bitsPerPixel != pDstDrawable->bitsPerPixel)
 +		goto fallback;
 +
 +	    if (pSrcDrawable->bitsPerPixel < 8)
 +		goto fallback;
 +
 +	    if (pGC && !(pGC->alu == GXcopy && EXA_PM_IS_SOLID(pSrcDrawable,  pGC->planemask)))
 +		goto fallback;
 +
 +	    while (nbox--)
 +	    {
 +		src = pSrcExaPixmap->sys_ptr + (pbox->y1 + dy + src_off_y) * src_stride + (pbox->x1 + dx + src_off_x) * (bpp / 8);
 +		if (!pExaScr->info->UploadToScreen(pDstPixmap, pbox->x1 + dst_off_x,
 +				pbox->y1 + dst_off_y, pbox->x2 - pbox->x1, pbox->y2 - pbox->y1,
 +				(char *) src, src_stride))
 +		    goto fallback;
 +
 +		pbox++;
 +	    }
 +	} else
 +	    goto fallback;
 +    } else
 +	goto fallback;
 +
 +    goto out;
 +
 +fallback:
 +    ret = FALSE;
 +
 +out:
 +    if (dstregion) {
 +	REGION_UNINIT(pScreen, dstregion);
 +	REGION_DESTROY(pScreen, dstregion);
 +    }
 +    if (srcregion) {
 +	REGION_UNINIT(pScreen, srcregion);
 +	REGION_DESTROY(pScreen, srcregion);
 +    }
 +
 +    return ret;
 +}
 +
 +void
 +exaCopyNtoN (DrawablePtr    pSrcDrawable,
 +	     DrawablePtr    pDstDrawable,
 +	     GCPtr	    pGC,
 +	     BoxPtr	    pbox,
 +	     int	    nbox,
 +	     int	    dx,
 +	     int	    dy,
 +	     Bool	    reverse,
 +	     Bool	    upsidedown,
 +	     Pixel	    bitplane,
 +	     void	    *closure)
 +{
 +    ExaScreenPriv(pDstDrawable->pScreen);
 +
 +    if (pExaScr->fallback_counter ||
 +	    (pExaScr->fallback_flags & EXA_FALLBACK_COPYWINDOW))
 +	return;
 +
 +    if (exaHWCopyNtoN(pSrcDrawable, pDstDrawable, pGC, pbox, nbox, dx, dy, reverse, upsidedown))
 +	return;
 +
 +    /* This is a CopyWindow, it's cleaner to fallback at the original call. */
 +    if (pExaScr->fallback_flags & EXA_ACCEL_COPYWINDOW) {
 +	pExaScr->fallback_flags |= EXA_FALLBACK_COPYWINDOW;
 +	return;
 +    }
 +
 +    /* fallback */
 +    ExaCheckCopyNtoN(pSrcDrawable, pDstDrawable, pGC, pbox, nbox, dx, dy, reverse, upsidedown, bitplane, closure);
 +}
 +
 +RegionPtr
 +exaCopyArea(DrawablePtr pSrcDrawable, DrawablePtr pDstDrawable, GCPtr pGC,
 +	    int srcx, int srcy, int width, int height, int dstx, int dsty)
 +{
 +    ExaScreenPriv (pDstDrawable->pScreen);
 +
 +    if (pExaScr->fallback_counter || pExaScr->swappedOut) {
 +        return  ExaCheckCopyArea(pSrcDrawable, pDstDrawable, pGC,
 +                                 srcx, srcy, width, height, dstx, dsty);
 +    }
 +
 +    return  miDoCopy (pSrcDrawable, pDstDrawable, pGC,
 +                      srcx, srcy, width, height,
 +                      dstx, dsty, exaCopyNtoN, 0, NULL);
 +}
 +
 +static void
 +exaPolyPoint(DrawablePtr pDrawable, GCPtr pGC, int mode, int npt,
 +	     DDXPointPtr ppt)
 +{
 +    ExaScreenPriv (pDrawable->pScreen);
 +    int i;
 +    xRectangle *prect;
 +
 +    /* If we can't reuse the current GC as is, don't bother accelerating the
 +     * points.
 +     */
 +    if (pExaScr->fallback_counter || pGC->fillStyle != FillSolid) {
 +	ExaCheckPolyPoint(pDrawable, pGC, mode, npt, ppt);
 +	return;
 +    }
 +
 +    prect = xalloc(sizeof(xRectangle) * npt);
 +    for (i = 0; i < npt; i++) {
 +	prect[i].x = ppt[i].x;
 +	prect[i].y = ppt[i].y;
 +	if (i > 0 && mode == CoordModePrevious) {
 +	    prect[i].x += prect[i - 1].x;
 +	    prect[i].y += prect[i - 1].y;
 +	}
 +	prect[i].width = 1;
 +	prect[i].height = 1;
 +    }
 +    pGC->ops->PolyFillRect(pDrawable, pGC, npt, prect);
 +    xfree(prect);
 +}
 +
 +/**
 + * exaPolylines() checks if it can accelerate the lines as a group of
 + * horizontal or vertical lines (rectangles), and uses existing rectangle fill
 + * acceleration if so.
 + */
 +static void
 +exaPolylines(DrawablePtr pDrawable, GCPtr pGC, int mode, int npt,
 +	     DDXPointPtr ppt)
 +{
 +    ExaScreenPriv (pDrawable->pScreen);
 +    xRectangle *prect;
 +    int x1, x2, y1, y2;
 +    int i;
 +
 +    if (pExaScr->fallback_counter) {
 +	ExaCheckPolylines(pDrawable, pGC, mode, npt, ppt);
 +	return;
 +    }
 +
 +    /* Don't try to do wide lines or non-solid fill style. */
 +    if (pGC->lineWidth != 0 || pGC->lineStyle != LineSolid ||
 +	pGC->fillStyle != FillSolid) {
 +	ExaCheckPolylines(pDrawable, pGC, mode, npt, ppt);
 +	return;
 +    }
 +
 +    prect = xalloc(sizeof(xRectangle) * (npt - 1));
 +    x1 = ppt[0].x;
 +    y1 = ppt[0].y;
 +    /* If we have any non-horizontal/vertical, fall back. */
 +    for (i = 0; i < npt - 1; i++) {
 +	if (mode == CoordModePrevious) {
 +	    x2 = x1 + ppt[i + 1].x;
 +	    y2 = y1 + ppt[i + 1].y;
 +	} else {
 +	    x2 = ppt[i + 1].x;
 +	    y2 = ppt[i + 1].y;
 +	}
 +
 +	if (x1 != x2 && y1 != y2) {
 +	    xfree(prect);
 +	    ExaCheckPolylines(pDrawable, pGC, mode, npt, ppt);
 +	    return;
 +	}
 +
 +	if (x1 < x2) {
 +	    prect[i].x = x1;
 +	    prect[i].width = x2 - x1 + 1;
 +	} else {
 +	    prect[i].x = x2;
 +	    prect[i].width = x1 - x2 + 1;
 +	}
 +	if (y1 < y2) {
 +	    prect[i].y = y1;
 +	    prect[i].height = y2 - y1 + 1;
 +	} else {
 +	    prect[i].y = y2;
 +	    prect[i].height = y1 - y2 + 1;
 +	}
 +
 +	x1 = x2;
 +	y1 = y2;
 +    }
 +    pGC->ops->PolyFillRect(pDrawable, pGC, npt - 1, prect);
 +    xfree(prect);
 +}
 +
 +/**
 + * exaPolySegment() checks if it can accelerate the lines as a group of
 + * horizontal or vertical lines (rectangles), and uses existing rectangle fill
 + * acceleration if so.
 + */
 +static void
 +exaPolySegment (DrawablePtr pDrawable, GCPtr pGC, int nseg,
 +		xSegment *pSeg)
 +{
 +    ExaScreenPriv (pDrawable->pScreen);
 +    xRectangle *prect;
 +    int i;
 +
 +    /* Don't try to do wide lines or non-solid fill style. */
 +    if (pExaScr->fallback_counter || pGC->lineWidth != 0 ||
 +	pGC->lineStyle != LineSolid || pGC->fillStyle != FillSolid)
 +    {
 +	ExaCheckPolySegment(pDrawable, pGC, nseg, pSeg);
 +	return;
 +    }
 +
 +    /* If we have any non-horizontal/vertical, fall back. */
 +    for (i = 0; i < nseg; i++) {
 +	if (pSeg[i].x1 != pSeg[i].x2 && pSeg[i].y1 != pSeg[i].y2) {
 +	    ExaCheckPolySegment(pDrawable, pGC, nseg, pSeg);
 +	    return;
 +	}
 +    }
 +
 +    prect = xalloc(sizeof(xRectangle) * nseg);
 +    for (i = 0; i < nseg; i++) {
 +	if (pSeg[i].x1 < pSeg[i].x2) {
 +	    prect[i].x = pSeg[i].x1;
 +	    prect[i].width = pSeg[i].x2 - pSeg[i].x1 + 1;
 +	} else {
 +	    prect[i].x = pSeg[i].x2;
 +	    prect[i].width = pSeg[i].x1 - pSeg[i].x2 + 1;
 +	}
 +	if (pSeg[i].y1 < pSeg[i].y2) {
 +	    prect[i].y = pSeg[i].y1;
 +	    prect[i].height = pSeg[i].y2 - pSeg[i].y1 + 1;
 +	} else {
 +	    prect[i].y = pSeg[i].y2;
 +	    prect[i].height = pSeg[i].y1 - pSeg[i].y2 + 1;
 +	}
 +
 +	/* don't paint last pixel */
 +	if (pGC->capStyle == CapNotLast) {
 +	    if (prect[i].width == 1)
 +		prect[i].height--;
 +	    else
 +		prect[i].width--;
 +	}
 +    }
 +    pGC->ops->PolyFillRect(pDrawable, pGC, nseg, prect);
 +    xfree(prect);
 +}
 +
 +static Bool exaFillRegionSolid (DrawablePtr pDrawable, RegionPtr pRegion,
 +				Pixel pixel, CARD32 planemask, CARD32 alu,
 +				unsigned int clientClipType);
 +
 +static void
 +exaPolyFillRect(DrawablePtr pDrawable,
 +		GCPtr	    pGC,
 +		int	    nrect,
 +		xRectangle  *prect)
 +{
 +    ExaScreenPriv (pDrawable->pScreen);
 +    RegionPtr	    pClip = fbGetCompositeClip(pGC);
 +    PixmapPtr	    pPixmap = exaGetDrawablePixmap(pDrawable);
 +    ExaPixmapPriv (pPixmap);
 +    register BoxPtr pbox;
 +    BoxPtr	    pextent;
 +    int		    extentX1, extentX2, extentY1, extentY2;
 +    int		    fullX1, fullX2, fullY1, fullY2;
 +    int		    partX1, partX2, partY1, partY2;
 +    int		    xoff, yoff;
 +    int		    xorg, yorg;
 +    int		    n;
 +    RegionPtr pReg = RECTS_TO_REGION(pScreen, nrect, prect, CT_UNSORTED);
 +
 +    /* Compute intersection of rects and clip region */
 +    REGION_TRANSLATE(pScreen, pReg, pDrawable->x, pDrawable->y);
 +    REGION_INTERSECT(pScreen, pReg, pClip, pReg);
 +
 +    if (!REGION_NUM_RECTS(pReg)) {
 +	goto out;
 +    }
 +
 +    exaGetDrawableDeltas(pDrawable, pPixmap, &xoff, &yoff);
 +
 +    if (pExaScr->fallback_counter || pExaScr->swappedOut ||
 +	    pExaPixmap->accel_blocked)
 +    {
 +	goto fallback;
 +    }
 +
 +    /* For ROPs where overlaps don't matter, convert rectangles to region and
 +     * call exaFillRegion{Solid,Tiled}.
 +     */
 +    if ((pGC->fillStyle == FillSolid || pGC->fillStyle == FillTiled) &&
 +	(nrect == 1 || pGC->alu == GXcopy || pGC->alu == GXclear ||
 +	 pGC->alu == GXnoop || pGC->alu == GXcopyInverted ||
 +	 pGC->alu == GXset)) {
 +	if (((pGC->fillStyle == FillSolid || pGC->tileIsPixel) &&
 +	     exaFillRegionSolid(pDrawable, pReg, pGC->fillStyle == FillSolid ?
 +				pGC->fgPixel : pGC->tile.pixel,	pGC->planemask,
 +				pGC->alu, pGC->clientClipType)) ||
 +	    (pGC->fillStyle == FillTiled && !pGC->tileIsPixel &&
 +	     exaFillRegionTiled(pDrawable, pReg, pGC->tile.pixmap, &pGC->patOrg,
 +				pGC->planemask, pGC->alu,
 +				pGC->clientClipType))) {
 +	    goto out;
 +	}
 +    }
 +
 +    if (pGC->fillStyle != FillSolid &&
 +	!(pGC->tileIsPixel && pGC->fillStyle == FillTiled))
 +    {
 +	goto fallback;
 +    }
 +
 +    if (pExaScr->do_migration) {
 +	ExaMigrationRec pixmaps[1];
 +
 +	pixmaps[0].as_dst = TRUE;
 +	pixmaps[0].as_src = FALSE;
 +	pixmaps[0].pPix = pPixmap;
 +	pixmaps[0].pReg = NULL;
 +
 +	exaDoMigration (pixmaps, 1, TRUE);
 +    }
 +
 +    if (!exaPixmapHasGpuCopy (pPixmap) ||
 +	!(*pExaScr->info->PrepareSolid) (pPixmap,
 +					 pGC->alu,
 +					 pGC->planemask,
 +					 pGC->fgPixel))
 +    {
 +fallback:
 +	ExaCheckPolyFillRect (pDrawable, pGC, nrect, prect);
 +	goto out;
 +    }
 +
 +    xorg = pDrawable->x;
 +    yorg = pDrawable->y;
 +
 +    pextent = REGION_EXTENTS(pGC->pScreen, pClip);
 +    extentX1 = pextent->x1;
 +    extentY1 = pextent->y1;
 +    extentX2 = pextent->x2;
 +    extentY2 = pextent->y2;
 +    while (nrect--)
 +    {
 +	fullX1 = prect->x + xorg;
 +	fullY1 = prect->y + yorg;
 +	fullX2 = fullX1 + (int) prect->width;
 +	fullY2 = fullY1 + (int) prect->height;
 +	prect++;
 +
 +	if (fullX1 < extentX1)
 +	    fullX1 = extentX1;
 +
 +	if (fullY1 < extentY1)
 +	    fullY1 = extentY1;
 +
 +	if (fullX2 > extentX2)
 +	    fullX2 = extentX2;
 +
 +	if (fullY2 > extentY2)
 +	    fullY2 = extentY2;
 +
 +	if ((fullX1 >= fullX2) || (fullY1 >= fullY2))
 +	    continue;
 +	n = REGION_NUM_RECTS (pClip);
 +	if (n == 1)
 +	{
 +	    (*pExaScr->info->Solid) (pPixmap,
 +				     fullX1 + xoff, fullY1 + yoff,
 +				     fullX2 + xoff, fullY2 + yoff);
 +	}
 +	else
 +	{
 +	    pbox = REGION_RECTS(pClip);
 +	    /*
 +	     * clip the rectangle to each box in the clip region
 +	     * this is logically equivalent to calling Intersect(),
 +	     * but rectangles may overlap each other here.
 +	     */
 +	    while(n--)
 +	    {
 +		partX1 = pbox->x1;
 +		if (partX1 < fullX1)
 +		    partX1 = fullX1;
 +		partY1 = pbox->y1;
 +		if (partY1 < fullY1)
 +		    partY1 = fullY1;
 +		partX2 = pbox->x2;
 +		if (partX2 > fullX2)
 +		    partX2 = fullX2;
 +		partY2 = pbox->y2;
 +		if (partY2 > fullY2)
 +		    partY2 = fullY2;
 +
 +		pbox++;
 +
 +		if (partX1 < partX2 && partY1 < partY2) {
 +		    (*pExaScr->info->Solid) (pPixmap,
 +					     partX1 + xoff, partY1 + yoff,
 +					     partX2 + xoff, partY2 + yoff);
 +		}
 +	    }
 +	}
 +    }
 +    (*pExaScr->info->DoneSolid) (pPixmap);
 +    exaMarkSync(pDrawable->pScreen);
 +
 +out:
 +    REGION_UNINIT(pScreen, pReg);
 +    REGION_DESTROY(pScreen, pReg);
 +}
 +
 +const GCOps exaOps = {
 +    exaFillSpans,
 +    ExaCheckSetSpans,
 +    exaPutImage,
 +    exaCopyArea,
 +    ExaCheckCopyPlane,
 +    exaPolyPoint,
 +    exaPolylines,
 +    exaPolySegment,
 +    miPolyRectangle,
 +    ExaCheckPolyArc,
 +    miFillPolygon,
 +    exaPolyFillRect,
 +    miPolyFillArc,
 +    miPolyText8,
 +    miPolyText16,
 +    miImageText8,
 +    miImageText16,
 +    ExaCheckImageGlyphBlt,
 +    ExaCheckPolyGlyphBlt,
 +    ExaCheckPushPixels,
 +};
 +
 +void
 +exaCopyWindow(WindowPtr pWin, DDXPointRec ptOldOrg, RegionPtr prgnSrc)
 +{
 +    RegionRec	rgnDst;
 +    int		dx, dy;
 +    PixmapPtr	pPixmap = (*pWin->drawable.pScreen->GetWindowPixmap) (pWin);
 +    ExaScreenPriv(pWin->drawable.pScreen);
 +
 +    dx = ptOldOrg.x - pWin->drawable.x;
 +    dy = ptOldOrg.y - pWin->drawable.y;
 +    REGION_TRANSLATE(pWin->drawable.pScreen, prgnSrc, -dx, -dy);
 +
 +    REGION_INIT (pWin->drawable.pScreen, &rgnDst, NullBox, 0);
 +
 +    REGION_INTERSECT(pWin->drawable.pScreen, &rgnDst, &pWin->borderClip, prgnSrc);
 +#ifdef COMPOSITE
 +    if (pPixmap->screen_x || pPixmap->screen_y)
 +	REGION_TRANSLATE (pWin->drawable.pScreen, &rgnDst,
 +			  -pPixmap->screen_x, -pPixmap->screen_y);
 +#endif
 +
 +    if (pExaScr->fallback_counter) {
 +	pExaScr->fallback_flags |= EXA_FALLBACK_COPYWINDOW;
 +	goto fallback;
 +    }
 +
 +    pExaScr->fallback_flags |= EXA_ACCEL_COPYWINDOW;
 +    miCopyRegion (&pPixmap->drawable, &pPixmap->drawable,
 +		  NULL,
 +		  &rgnDst, dx, dy, exaCopyNtoN, 0, NULL);
 +    pExaScr->fallback_flags &= ~EXA_ACCEL_COPYWINDOW;
 +
 +fallback:
 +    REGION_UNINIT(pWin->drawable.pScreen, &rgnDst);
 +
 +    if (pExaScr->fallback_flags & EXA_FALLBACK_COPYWINDOW) {
 +	pExaScr->fallback_flags &= ~EXA_FALLBACK_COPYWINDOW;
 +	REGION_TRANSLATE(pWin->drawable.pScreen, prgnSrc, dx, dy);
 +	ExaCheckCopyWindow(pWin, ptOldOrg, prgnSrc);
 +    }
 +}
 +
 +static Bool
 +exaFillRegionSolid (DrawablePtr	pDrawable, RegionPtr pRegion, Pixel pixel,
 +		    CARD32 planemask, CARD32 alu, unsigned int clientClipType)
 +{
 +    ExaScreenPriv(pDrawable->pScreen);
 +    PixmapPtr pPixmap = exaGetDrawablePixmap (pDrawable);
 +    ExaPixmapPriv (pPixmap);
 +    int xoff, yoff;
 +    Bool ret = FALSE;
 +
 +    exaGetDrawableDeltas(pDrawable, pPixmap, &xoff, &yoff);
 +    REGION_TRANSLATE(pScreen, pRegion, xoff, yoff);
 +
 +    if (pExaScr->fallback_counter || pExaPixmap->accel_blocked)
 +	goto out;
 +
 +    if (pExaScr->do_migration) {
 +	ExaMigrationRec pixmaps[1];
 +
 +	pixmaps[0].as_dst = TRUE;
 +	pixmaps[0].as_src = FALSE;
 +	pixmaps[0].pPix = pPixmap;
 +	pixmaps[0].pReg = exaGCReadsDestination(pDrawable, planemask, FillSolid,
 +						alu, clientClipType) ? NULL : pRegion;
 +
 +	exaDoMigration (pixmaps, 1, TRUE);
 +    }
 +
 +    if (exaPixmapHasGpuCopy (pPixmap) &&
 +	(*pExaScr->info->PrepareSolid) (pPixmap, alu, planemask, pixel))
 +    {
 +	int nbox;
 +	BoxPtr pBox;
 +
 +	nbox = REGION_NUM_RECTS (pRegion);
 +	pBox = REGION_RECTS (pRegion);
 +
 +	while (nbox--)
 +	{
 +	    (*pExaScr->info->Solid) (pPixmap, pBox->x1, pBox->y1, pBox->x2,
 +				     pBox->y2);
 +	    pBox++;
 +	}
 +	(*pExaScr->info->DoneSolid) (pPixmap);
 +	exaMarkSync(pDrawable->pScreen);
 +
 +	if (pExaPixmap->pDamage &&
 +	    pExaPixmap->sys_ptr && pDrawable->type == DRAWABLE_PIXMAP &&
 +	    pDrawable->width == 1 && pDrawable->height == 1 &&
 +	    pDrawable->bitsPerPixel != 24) {
 +	    ExaPixmapPriv(pPixmap);
 +	    RegionPtr pending_damage = DamagePendingRegion(pExaPixmap->pDamage);
 +
 +	    switch (pDrawable->bitsPerPixel) {
 +	    case 32:
 +		*(CARD32*)pExaPixmap->sys_ptr = pixel;
 +		break;
 +	    case 16:
 +		*(CARD16*)pExaPixmap->sys_ptr = pixel;
 +		break;
 +	    case 8:
 +		*(CARD8*)pExaPixmap->sys_ptr = pixel;
 +	    }
 +
 +	    REGION_UNION(pScreen, &pExaPixmap->validSys, &pExaPixmap->validSys,
 +			 pRegion);
 +	    REGION_UNION(pScreen, &pExaPixmap->validFB, &pExaPixmap->validFB,
 +			 pRegion);
 +	    REGION_SUBTRACT(pScreen, pending_damage, pending_damage, pRegion);
 +	}
 +
 +	ret = TRUE;
 +    }
 +
 +out:
 +    REGION_TRANSLATE(pScreen, pRegion, -xoff, -yoff);
 +
 +    return ret;
 +}
 +
 +/* Try to do an accelerated tile of the pTile into pRegion of pDrawable.
 + * Based on fbFillRegionTiled(), fbTile().
 + */
 +Bool
 +exaFillRegionTiled (DrawablePtr pDrawable, RegionPtr pRegion, PixmapPtr pTile,
 +		    DDXPointPtr pPatOrg, CARD32 planemask, CARD32 alu,
 +		    unsigned int clientClipType)
 +{
 +    ExaScreenPriv(pDrawable->pScreen);
 +    PixmapPtr pPixmap;
 +    ExaPixmapPrivPtr pExaPixmap;
 +    ExaPixmapPrivPtr pTileExaPixmap = ExaGetPixmapPriv(pTile);
 +    int xoff, yoff;
 +    int tileWidth, tileHeight;
 +    int nbox = REGION_NUM_RECTS (pRegion);
 +    BoxPtr pBox = REGION_RECTS (pRegion);
 +    Bool ret = FALSE;
 +    int i;
 +
 +    tileWidth = pTile->drawable.width;
 +    tileHeight = pTile->drawable.height;
 +
 +    /* If we're filling with a solid color, grab it out and go to
 +     * FillRegionSolid, saving numerous copies.
 +     */
 +    if (tileWidth == 1 && tileHeight == 1)
 +	return exaFillRegionSolid(pDrawable, pRegion,
 +				  exaGetPixmapFirstPixel (pTile), planemask,
 +				  alu, clientClipType);
 +
 +    pPixmap = exaGetDrawablePixmap (pDrawable);
 +    pExaPixmap = ExaGetPixmapPriv (pPixmap);
 +
 +    if (pExaScr->fallback_counter || pExaPixmap->accel_blocked ||
 +	    pTileExaPixmap->accel_blocked)
 +	return FALSE;
 +
 +    if (pExaScr->do_migration) {
 +	ExaMigrationRec pixmaps[2];
 +
 +	pixmaps[0].as_dst = TRUE;
 +	pixmaps[0].as_src = FALSE;
 +	pixmaps[0].pPix = pPixmap;
 +	pixmaps[0].pReg = exaGCReadsDestination(pDrawable, planemask, FillTiled,
 +						alu, clientClipType) ? NULL : pRegion;
 +	pixmaps[1].as_dst = FALSE;
 +	pixmaps[1].as_src = TRUE;
 +	pixmaps[1].pPix = pTile;
 +	pixmaps[1].pReg = NULL;
 +
 +	exaDoMigration (pixmaps, 2, TRUE);
 +    }
 +
 +    pPixmap = exaGetOffscreenPixmap (pDrawable, &xoff, &yoff);
 +
 +    if (!pPixmap || !exaPixmapHasGpuCopy(pTile))
 +	return FALSE;
 +
 +    if ((*pExaScr->info->PrepareCopy) (pTile, pPixmap, 1, 1, alu, planemask))
 +    {
 +	if (xoff || yoff)
 +	    REGION_TRANSLATE(pScreen, pRegion, xoff, yoff);
 +
 +	for (i = 0; i < nbox; i++)
 +	{
 +	    int height = pBox[i].y2 - pBox[i].y1;
 +	    int dstY = pBox[i].y1;
 +	    int tileY;
 +
 +	    if (alu == GXcopy)
 +		height = min(height, tileHeight);
 +
 +	    modulus(dstY - yoff - pDrawable->y - pPatOrg->y, tileHeight, tileY);
 +
 +	    while (height > 0) {
 +		int width = pBox[i].x2 - pBox[i].x1;
 +		int dstX = pBox[i].x1;
 +		int tileX;
 +		int h = tileHeight - tileY;
 +
 +		if (alu == GXcopy)
 +		    width = min(width, tileWidth);
 +
 +		if (h > height)
 +		    h = height;
 +		height -= h;
 +
 +		modulus(dstX - xoff - pDrawable->x - pPatOrg->x, tileWidth,
 +			tileX);
 +
 +		while (width > 0) {
 +		    int w = tileWidth - tileX;
 +		    if (w > width)
 +			w = width;
 +		    width -= w;
 +
 +		    (*pExaScr->info->Copy) (pPixmap, tileX, tileY, dstX, dstY,
 +					    w, h);
 +		    dstX += w;
 +		    tileX = 0;
 +		}
 +		dstY += h;
 +		tileY = 0;
 +	    }
 +	}
 +	(*pExaScr->info->DoneCopy) (pPixmap);
 +
 +	/* With GXcopy, we only need to do the basic algorithm up to the tile
 +	 * size; then, we can just keep doubling the destination in each
 +	 * direction until it fills the box. This way, the number of copy
 +	 * operations is O(log(rx)) + O(log(ry)) instead of O(rx * ry), where
 +	 * rx/ry is the ratio between box and tile width/height. This can make
 +	 * a big difference if each driver copy incurs a significant constant
 +	 * overhead.
 +	 */
 +	if (alu != GXcopy)
 +	    ret = TRUE;
 +	else {
 +	    Bool more_copy = FALSE;
 +
 +	    for (i = 0; i < nbox; i++) {
 +		int dstX = pBox[i].x1 + tileWidth;
 +		int dstY = pBox[i].y1 + tileHeight;
 +
 +		if ((dstX < pBox[i].x2) || (dstY < pBox[i].y2)) {
 +		    more_copy = TRUE;
 +		    break;
 +		}
 +	    }
 +
 +	    if (more_copy == FALSE)
 +		ret = TRUE;
 +
 +	    if (more_copy && (*pExaScr->info->PrepareCopy) (pPixmap, pPixmap,
 +							    1, 1, alu, planemask)) {
 +		for (i = 0; i < nbox; i++)
 +		{
 +		    int dstX = pBox[i].x1 + tileWidth;
 +		    int dstY = pBox[i].y1 + tileHeight;
 +		    int width = min(pBox[i].x2 - dstX, tileWidth);
 +		    int height = min(pBox[i].y2 - pBox[i].y1, tileHeight);
 +
 +		    while (dstX < pBox[i].x2) {
 +			(*pExaScr->info->Copy) (pPixmap, pBox[i].x1, pBox[i].y1,
 +						dstX, pBox[i].y1, width, height);
 +			dstX += width;
 +			width = min(pBox[i].x2 - dstX, width * 2);
 +		    }
 +
 +		    width = pBox[i].x2 - pBox[i].x1;
 +		    height = min(pBox[i].y2 - dstY, tileHeight);
 +
 +		    while (dstY < pBox[i].y2) {
 +			(*pExaScr->info->Copy) (pPixmap, pBox[i].x1, pBox[i].y1,
 +						pBox[i].x1, dstY, width, height);
 +			dstY += height;
 +			height = min(pBox[i].y2 - dstY, height * 2);
 +		    }
 +		}
 +
 +		(*pExaScr->info->DoneCopy) (pPixmap);
 +
 +		ret = TRUE;
 +	    }
 +	}
 +
 +	exaMarkSync(pDrawable->pScreen);
 +
 +	if (xoff || yoff)
 +	    REGION_TRANSLATE(pScreen, pRegion, -xoff, -yoff);
 +    }
 +
 +    return ret;
 +}
 +
 +
 +/**
 + * Accelerates GetImage for solid ZPixmap downloads from framebuffer memory.
 + *
 + * This is probably the only case we actually care about.  The rest fall through
 + * to migration and fbGetImage, which hopefully will result in migration pushing
 + * the pixmap out of framebuffer.
 + */
 +void
 +exaGetImage (DrawablePtr pDrawable, int x, int y, int w, int h,
 +	     unsigned int format, unsigned long planeMask, char *d)
 +{
 +    ExaScreenPriv (pDrawable->pScreen);
 +    PixmapPtr pPix = exaGetDrawablePixmap (pDrawable);
 +    ExaPixmapPriv(pPix);
 +    int xoff, yoff;
 +    Bool ok;
 +
 +    if (pExaScr->fallback_counter || pExaScr->swappedOut)
 +	goto fallback;
 +
 +    /* If there's a system copy, we want to save the result there */
 +    if (pExaPixmap->pDamage)
 +	goto fallback;
 +
 +    pPix = exaGetOffscreenPixmap (pDrawable, &xoff, &yoff);
 +
 +    if (pPix == NULL || pExaScr->info->DownloadFromScreen == NULL)
 +	goto fallback;
 +
 +    /* Only cover the ZPixmap, solid copy case. */
 +    if (format != ZPixmap || !EXA_PM_IS_SOLID(pDrawable, planeMask))
 +	goto fallback;
 +
 +    /* Only try to handle the 8bpp and up cases, since we don't want to think
 +     * about <8bpp.
 +     */
 +    if (pDrawable->bitsPerPixel < 8)
 +	goto fallback;
 +
 +    ok = pExaScr->info->DownloadFromScreen(pPix, pDrawable->x + x + xoff,
 +					   pDrawable->y + y + yoff, w, h, d,
 +					   PixmapBytePad(w, pDrawable->depth));
 +    if (ok) {
 +	exaWaitSync(pDrawable->pScreen);
 +	return;
 +    }
 +
 +fallback:
 +    ExaCheckGetImage(pDrawable, x, y, w, h, format, planeMask, d);
 +}
 diff --git a/xorg-server/exa/exa_migration_classic.c b/xorg-server/exa/exa_migration_classic.c index 871679ffc..9e1626c69 100644 --- a/xorg-server/exa/exa_migration_classic.c +++ b/xorg-server/exa/exa_migration_classic.c @@ -1,745 +1,745 @@ -/* - * Copyright © 2006 Intel Corporation - * - * Permission is hereby granted, free of charge, to any person obtaining a - * copy of this software and associated documentation files (the "Software"), - * to deal in the Software without restriction, including without limitation - * the rights to use, copy, modify, merge, publish, distribute, sublicense, - * and/or sell copies of the Software, and to permit persons to whom the - * Software is furnished to do so, subject to the following conditions: - * - * The above copyright notice and this permission notice (including the next - * paragraph) shall be included in all copies or substantial portions of the - * Software. - * - * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR - * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, - * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL - * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER - * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, - * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE - * SOFTWARE. - * - * Authors: - *    Eric Anholt <eric@anholt.net> - *    Michel Dänzer <michel@tungstengraphics.com> - * - */ - -#ifdef HAVE_DIX_CONFIG_H -#include <dix-config.h> -#endif - -#include <string.h> - -#include "exa_priv.h" -#include "exa.h" - -#if DEBUG_MIGRATE -#define DBG_MIGRATE(a) ErrorF a -#else -#define DBG_MIGRATE(a) -#endif - -/** - * The fallback path for UTS/DFS failing is to just memcpy.  exaCopyDirtyToSys - * and exaCopyDirtyToFb both needed to do this loop. - */ -static void -exaMemcpyBox (PixmapPtr pPixmap, BoxPtr pbox, CARD8 *src, int src_pitch, -	      CARD8 *dst, int dst_pitch) - { -    int i, cpp = pPixmap->drawable.bitsPerPixel / 8; -    int bytes = (pbox->x2 - pbox->x1) * cpp; - -    src += pbox->y1 * src_pitch + pbox->x1 * cpp; -    dst += pbox->y1 * dst_pitch + pbox->x1 * cpp; - -    for (i = pbox->y2 - pbox->y1; i; i--) { -	memcpy (dst, src, bytes); -	src += src_pitch; -	dst += dst_pitch; -    } -} - -/** - * Returns TRUE if the pixmap is dirty (has been modified in its current - * location compared to the other), or lacks a private for tracking - * dirtiness. - */ -static Bool -exaPixmapIsDirty (PixmapPtr pPix) -{ -    ExaPixmapPriv (pPix); - -    if (pExaPixmap == NULL) -	EXA_FatalErrorDebugWithRet(("EXA bug: exaPixmapIsDirty was called on a non-exa pixmap.\n"), TRUE); - -    if (!pExaPixmap->pDamage) -	return FALSE; - -    return REGION_NOTEMPTY (pScreen, DamageRegion(pExaPixmap->pDamage)) || -	!REGION_EQUAL(pScreen, &pExaPixmap->validSys, &pExaPixmap->validFB); -} - -/** - * Returns TRUE if the pixmap is either pinned in FB, or has a sufficient score - * to be considered "should be in framebuffer".  That's just anything that has - * had more acceleration than fallbacks, or has no score yet. - * - * Only valid if using a migration scheme that tracks score. - */ -static Bool -exaPixmapShouldBeInFB (PixmapPtr pPix) -{ -    ExaPixmapPriv (pPix); - -    if (exaPixmapIsPinned (pPix)) -	return TRUE; - -    return pExaPixmap->score >= 0; -} - -/** - * If the pixmap is currently dirty, this copies at least the dirty area from - * FB to system or vice versa.  Both areas must be allocated. - */ -static void -exaCopyDirty(ExaMigrationPtr migrate, RegionPtr pValidDst, RegionPtr pValidSrc, -	     Bool (*transfer) (PixmapPtr pPix, int x, int y, int w, int h, -			       char *sys, int sys_pitch), int fallback_index, -	     void (*sync) (ScreenPtr pScreen)) -{ -    PixmapPtr pPixmap = migrate->pPix; -    ExaPixmapPriv (pPixmap); -    RegionPtr damage = DamageRegion (pExaPixmap->pDamage); -    RegionRec CopyReg; -    Bool save_use_gpu_copy; -    int save_pitch; -    BoxPtr pBox; -    int nbox; -    Bool access_prepared = FALSE; -    Bool need_sync = FALSE; - -    /* Damaged bits are valid in current copy but invalid in other one */ -    if (pExaPixmap->use_gpu_copy) { -	REGION_UNION(pScreen, &pExaPixmap->validFB, &pExaPixmap->validFB, -		     damage); -	REGION_SUBTRACT(pScreen, &pExaPixmap->validSys, &pExaPixmap->validSys, -			damage); -    } else { -	REGION_UNION(pScreen, &pExaPixmap->validSys, &pExaPixmap->validSys, -		     damage); -	REGION_SUBTRACT(pScreen, &pExaPixmap->validFB, &pExaPixmap->validFB, -			damage); -    } - -    REGION_EMPTY(pScreen, damage); - -    /* Copy bits valid in source but not in destination */ -    REGION_NULL(pScreen, &CopyReg); -    REGION_SUBTRACT(pScreen, &CopyReg, pValidSrc, pValidDst); - -    if (migrate->as_dst) { -	ExaScreenPriv (pPixmap->drawable.pScreen); - -	/* XXX: The pending damage region will be marked as damaged after the -	 * operation, so it should serve as an upper bound for the region that -	 * needs to be synchronized for the operation. Unfortunately, this -	 * causes corruption in some cases, e.g. when starting compiz. See -	 * https://bugs.freedesktop.org/show_bug.cgi?id=12916 . -	 */ -	if (pExaScr->optimize_migration) { -	    RegionPtr pending_damage = DamagePendingRegion(pExaPixmap->pDamage); - -#if DEBUG_MIGRATE -	    if (REGION_NIL(pending_damage)) { -		static Bool firsttime = TRUE; - -		if (firsttime) { -		    ErrorF("%s: Pending damage region empty!\n", __func__); -		    firsttime = FALSE; -		} -	    } -#endif - -	    /* Try to prevent destination valid region from growing too many -	     * rects by filling it up to the extents of the union of the -	     * destination valid region and the pending damage region. -	     */ -	    if (REGION_NUM_RECTS(pValidDst) > 10) { -		BoxRec box; -		BoxPtr pValidExt, pDamageExt; -		RegionRec closure; - -		pValidExt = REGION_EXTENTS(pScreen, pValidDst); -		pDamageExt = REGION_EXTENTS(pScreen, pending_damage); - -		box.x1 = min(pValidExt->x1, pDamageExt->x1); -		box.y1 = min(pValidExt->y1, pDamageExt->y1); -		box.x2 = max(pValidExt->x2, pDamageExt->x2); -		box.y2 = max(pValidExt->y2, pDamageExt->y2); - -		REGION_INIT(pScreen, &closure, &box, 0); -		REGION_INTERSECT(pScreen, &CopyReg, &CopyReg, &closure); -	    } else -		REGION_INTERSECT(pScreen, &CopyReg, &CopyReg, pending_damage); -	} - -	/* The caller may provide a region to be subtracted from the calculated -	 * dirty region. This is to avoid migration of bits that don't -	 * contribute to the result of the operation. -	 */ -	if (migrate->pReg) -	    REGION_SUBTRACT(pScreen, &CopyReg, &CopyReg, migrate->pReg); -    } else { -	/* The caller may restrict the region to be migrated for source pixmaps -	 * to what's relevant for the operation. -	 */ -	if (migrate->pReg) -	    REGION_INTERSECT(pScreen, &CopyReg, &CopyReg, migrate->pReg); -    } - -    pBox = REGION_RECTS(&CopyReg); -    nbox = REGION_NUM_RECTS(&CopyReg); - -    save_use_gpu_copy = pExaPixmap->use_gpu_copy; -    save_pitch = pPixmap->devKind; -    pExaPixmap->use_gpu_copy = TRUE; -    pPixmap->devKind = pExaPixmap->fb_pitch; - -    while (nbox--) { -	pBox->x1 = max(pBox->x1, 0); -	pBox->y1 = max(pBox->y1, 0); -	pBox->x2 = min(pBox->x2, pPixmap->drawable.width); -	pBox->y2 = min(pBox->y2, pPixmap->drawable.height); - -	if (pBox->x1 >= pBox->x2 || pBox->y1 >= pBox->y2) -	    continue; - -	if (!transfer || !transfer (pPixmap, -				    pBox->x1, pBox->y1, -				    pBox->x2 - pBox->x1, -				    pBox->y2 - pBox->y1, -				    (char *) (pExaPixmap->sys_ptr -				    + pBox->y1 * pExaPixmap->sys_pitch -				    + pBox->x1 * pPixmap->drawable.bitsPerPixel / 8), -				    pExaPixmap->sys_pitch)) -	{ -	    if (!access_prepared) { -		ExaDoPrepareAccess(pPixmap, fallback_index); -		access_prepared = TRUE; -	    } -	    if (fallback_index == EXA_PREPARE_DEST) { -		exaMemcpyBox (pPixmap, pBox, -			      pExaPixmap->sys_ptr, pExaPixmap->sys_pitch, -			      pPixmap->devPrivate.ptr, pPixmap->devKind); -	    } else { -		exaMemcpyBox (pPixmap, pBox, -			      pPixmap->devPrivate.ptr, pPixmap->devKind, -			      pExaPixmap->sys_ptr, pExaPixmap->sys_pitch); -	    } -	} else -	    need_sync = TRUE; - -	pBox++; -    } - -    pExaPixmap->use_gpu_copy = save_use_gpu_copy; -    pPixmap->devKind = save_pitch; - -    /* Try to prevent source valid region from growing too many rects by -     * removing parts of it which are also in the destination valid region. -     * Removing anything beyond that would lead to data loss. -     */ -    if (REGION_NUM_RECTS(pValidSrc) > 20) -	REGION_SUBTRACT(pScreen, pValidSrc, pValidSrc, pValidDst); - -    /* The copied bits are now valid in destination */ -    REGION_UNION(pScreen, pValidDst, pValidDst, &CopyReg); - -    REGION_UNINIT(pScreen, &CopyReg); - -    if (access_prepared) -	exaFinishAccess(&pPixmap->drawable, fallback_index); -    else if (need_sync && sync) -	sync (pPixmap->drawable.pScreen); -} - -/** - * If the pixmap is currently dirty, this copies at least the dirty area from - * the framebuffer  memory copy to the system memory copy.  Both areas must be - * allocated. - */ -void -exaCopyDirtyToSys (ExaMigrationPtr migrate) -{ -    PixmapPtr pPixmap = migrate->pPix; -    ExaScreenPriv (pPixmap->drawable.pScreen); -    ExaPixmapPriv (pPixmap); - -    exaCopyDirty(migrate, &pExaPixmap->validSys, &pExaPixmap->validFB, -		 pExaScr->info->DownloadFromScreen, EXA_PREPARE_SRC, -		 exaWaitSync); -} - -/** - * If the pixmap is currently dirty, this copies at least the dirty area from - * the system memory copy to the framebuffer memory copy.  Both areas must be - * allocated. - */ -void -exaCopyDirtyToFb (ExaMigrationPtr migrate) -{ -    PixmapPtr pPixmap = migrate->pPix; -    ExaScreenPriv (pPixmap->drawable.pScreen); -    ExaPixmapPriv (pPixmap); - -    exaCopyDirty(migrate, &pExaPixmap->validFB, &pExaPixmap->validSys, -		 pExaScr->info->UploadToScreen, EXA_PREPARE_DEST, NULL); -} - -/** - * Allocates a framebuffer copy of the pixmap if necessary, and then copies - * any necessary pixmap data into the framebuffer copy and points the pixmap at - * it. - * - * Note that when first allocated, a pixmap will have FALSE dirty flag. - * This is intentional because pixmap data starts out undefined.  So if we move - * it in due to the first operation against it being accelerated, it will have - * undefined framebuffer contents that we didn't have to upload.  If we do - * moveouts (and moveins) after the first movein, then we will only have to copy - * back and forth if the pixmap was written to after the last synchronization of - * the two copies.  Then, at exaPixmapSave (when the framebuffer copy goes away) - * we mark the pixmap dirty, so that the next exaMoveInPixmap will actually move - * all the data, since it's almost surely all valid now. - */ -static void -exaDoMoveInPixmap (ExaMigrationPtr migrate) -{ -    PixmapPtr pPixmap = migrate->pPix; -    ScreenPtr pScreen = pPixmap->drawable.pScreen; -    ExaScreenPriv (pScreen); -    ExaPixmapPriv (pPixmap); - -    /* If we're VT-switched away, no touching card memory allowed. */ -    if (pExaScr->swappedOut) -	return; - -    /* If we're not allowed to move, then fail. */ -    if (exaPixmapIsPinned(pPixmap)) -	return; - -    /* Don't migrate in pixmaps which are less than 8bpp.  This avoids a lot of -     * fragility in EXA, and <8bpp is probably not used enough any more to care -     * (at least, not in acceleratd paths). -     */ -    if (pPixmap->drawable.bitsPerPixel < 8) -	return; - -    if (pExaPixmap->accel_blocked) -	return; - -    if (pExaPixmap->area == NULL) { -	pExaPixmap->area = -	    exaOffscreenAlloc (pScreen, pExaPixmap->fb_size, -			       pExaScr->info->pixmapOffsetAlign, FALSE, -                               exaPixmapSave, (pointer) pPixmap); -	if (pExaPixmap->area == NULL) -	    return; - -	pExaPixmap->fb_ptr = (CARD8 *) pExaScr->info->memoryBase + -				       pExaPixmap->area->offset; -    } - -    exaCopyDirtyToFb (migrate); - -    if (exaPixmapHasGpuCopy(pPixmap)) -	return; - -    DBG_MIGRATE (("-> %p (0x%x) (%dx%d) (%c)\n", pPixmap, -		  (ExaGetPixmapPriv(pPixmap)->area ? -                   ExaGetPixmapPriv(pPixmap)->area->offset : 0), -		  pPixmap->drawable.width, -		  pPixmap->drawable.height, -		  exaPixmapIsDirty(pPixmap) ? 'd' : 'c')); - -    pExaPixmap->use_gpu_copy = TRUE; - -    pPixmap->devKind = pExaPixmap->fb_pitch; -    pPixmap->drawable.serialNumber = NEXT_SERIAL_NUMBER; -} - -void -exaMoveInPixmap_classic (PixmapPtr pPixmap) -{ -    static ExaMigrationRec migrate = { .as_dst = FALSE, .as_src = TRUE, -				       .pReg = NULL }; - -    migrate.pPix = pPixmap; -    exaDoMoveInPixmap (&migrate); -} - -/** - * Switches the current active location of the pixmap to system memory, copying - * updated data out if necessary. - */ -static void -exaDoMoveOutPixmap (ExaMigrationPtr migrate) -{ -    PixmapPtr pPixmap = migrate->pPix; -    ExaPixmapPriv (pPixmap); - -    if (!pExaPixmap->area || exaPixmapIsPinned(pPixmap)) -	return; - -    exaCopyDirtyToSys (migrate); - -    if (exaPixmapHasGpuCopy(pPixmap)) { - -	DBG_MIGRATE (("<- %p (%p) (%dx%d) (%c)\n", pPixmap, -		      (void*)(ExaGetPixmapPriv(pPixmap)->area ? -			      ExaGetPixmapPriv(pPixmap)->area->offset : 0), -		      pPixmap->drawable.width, -		      pPixmap->drawable.height, -		      exaPixmapIsDirty(pPixmap) ? 'd' : 'c')); - -	pExaPixmap->use_gpu_copy = FALSE; - -	pPixmap->devKind = pExaPixmap->sys_pitch; -	pPixmap->drawable.serialNumber = NEXT_SERIAL_NUMBER; -    } -} - -void -exaMoveOutPixmap_classic (PixmapPtr pPixmap) -{ -    static ExaMigrationRec migrate = { .as_dst = FALSE, .as_src = TRUE, -				       .pReg = NULL }; - -    migrate.pPix = pPixmap; -    exaDoMoveOutPixmap (&migrate); -} - - -/** - * Copies out important pixmap data and removes references to framebuffer area. - * Called when the memory manager decides it's time to kick the pixmap out of - * framebuffer entirely. - */ -void -exaPixmapSave (ScreenPtr pScreen, ExaOffscreenArea *area) -{ -    PixmapPtr pPixmap = area->privData; -    ExaPixmapPriv(pPixmap); - -    exaMoveOutPixmap(pPixmap); - -    pExaPixmap->fb_ptr = NULL; -    pExaPixmap->area = NULL; - -    /* Mark all FB bits as invalid, so all valid system bits get copied to FB -     * next time */ -    REGION_EMPTY(pPixmap->drawable.pScreen, &pExaPixmap->validFB); -} - -/** - * For the "greedy" migration scheme, pushes the pixmap toward being located in - * framebuffer memory. - */ -static void -exaMigrateTowardFb (ExaMigrationPtr migrate) -{ -    PixmapPtr pPixmap = migrate->pPix; -    ExaPixmapPriv (pPixmap); - -    if (pExaPixmap->score == EXA_PIXMAP_SCORE_PINNED) { -	DBG_MIGRATE(("UseScreen: not migrating pinned pixmap %p\n", -		     (pointer)pPixmap)); -	return; -    } - -    DBG_MIGRATE(("UseScreen %p score %d\n", -		 (pointer)pPixmap, pExaPixmap->score)); - -    if (pExaPixmap->score == EXA_PIXMAP_SCORE_INIT) { -	exaDoMoveInPixmap(migrate); -	pExaPixmap->score = 0; -    } - -    if (pExaPixmap->score < EXA_PIXMAP_SCORE_MAX) -	pExaPixmap->score++; - -    if (pExaPixmap->score >= EXA_PIXMAP_SCORE_MOVE_IN && -	!exaPixmapHasGpuCopy(pPixmap)) -    { -	exaDoMoveInPixmap(migrate); -    } - -    if (exaPixmapHasGpuCopy(pPixmap)) { -	exaCopyDirtyToFb (migrate); -	ExaOffscreenMarkUsed (pPixmap); -    } else -	exaCopyDirtyToSys (migrate); -} - -/** - * For the "greedy" migration scheme, pushes the pixmap toward being located in - * system memory. - */ -static void -exaMigrateTowardSys (ExaMigrationPtr migrate) -{ -    PixmapPtr pPixmap = migrate->pPix; -    ExaPixmapPriv (pPixmap); - -    DBG_MIGRATE(("UseMem: %p score %d\n", (pointer)pPixmap, pExaPixmap->score)); - -    if (pExaPixmap->score == EXA_PIXMAP_SCORE_PINNED) -	return; - -    if (pExaPixmap->score == EXA_PIXMAP_SCORE_INIT) -	pExaPixmap->score = 0; - -    if (pExaPixmap->score > EXA_PIXMAP_SCORE_MIN) -	pExaPixmap->score--; - -    if (pExaPixmap->score <= EXA_PIXMAP_SCORE_MOVE_OUT && pExaPixmap->area) -	exaDoMoveOutPixmap(migrate); - -    if (exaPixmapHasGpuCopy(pPixmap)) { -	exaCopyDirtyToFb (migrate); -	ExaOffscreenMarkUsed (pPixmap); -    } else -	exaCopyDirtyToSys (migrate); -} - -/** - * If the pixmap has both a framebuffer and system memory copy, this function - * asserts that both of them are the same. - */ -static Bool -exaAssertNotDirty (PixmapPtr pPixmap) -{ -    ExaPixmapPriv (pPixmap); -    CARD8 *dst, *src; -    RegionRec ValidReg; -    int dst_pitch, src_pitch, cpp, y, nbox, save_pitch; -    BoxPtr pBox; -    Bool ret = TRUE, save_use_gpu_copy; - -    if (exaPixmapIsPinned(pPixmap) || pExaPixmap->area == NULL) -	return ret; - -    REGION_NULL(pScreen, &ValidReg); -    REGION_INTERSECT(pScreen, &ValidReg, &pExaPixmap->validFB, -		     &pExaPixmap->validSys); -    nbox = REGION_NUM_RECTS(&ValidReg); - -    if (!nbox) -	goto out; - -    pBox = REGION_RECTS(&ValidReg); - -    dst_pitch = pExaPixmap->sys_pitch; -    src_pitch = pExaPixmap->fb_pitch; -    cpp = pPixmap->drawable.bitsPerPixel / 8; - -    save_use_gpu_copy = pExaPixmap->use_gpu_copy; -    save_pitch = pPixmap->devKind; -    pExaPixmap->use_gpu_copy = TRUE; -    pPixmap->devKind = pExaPixmap->fb_pitch; - -    if (!ExaDoPrepareAccess(pPixmap, EXA_PREPARE_SRC)) -	goto skip; - -    while (nbox--) { -	    int rowbytes; - -	    pBox->x1 = max(pBox->x1, 0); -	    pBox->y1 = max(pBox->y1, 0); -	    pBox->x2 = min(pBox->x2, pPixmap->drawable.width); -	    pBox->y2 = min(pBox->y2, pPixmap->drawable.height); - -	    if (pBox->x1 >= pBox->x2 || pBox->y1 >= pBox->y2) -		continue; - -	    rowbytes = (pBox->x2 - pBox->x1) * cpp; -	    src = (CARD8 *) pPixmap->devPrivate.ptr + pBox->y1 * src_pitch + pBox->x1 * cpp; -	    dst = pExaPixmap->sys_ptr + pBox->y1 * dst_pitch + pBox->x1 * cpp; - -	    for (y = pBox->y1; y < pBox->y2; -		 y++, src += src_pitch, dst += dst_pitch) { -		if (memcmp(dst, src, rowbytes) != 0) { -		    ret = FALSE; -		    exaPixmapDirty(pPixmap, pBox->x1, pBox->y1, pBox->x2, -				   pBox->y2); -		    break; -		} -	    } -    } - -skip: -    exaFinishAccess(&pPixmap->drawable, EXA_PREPARE_SRC); - -    pExaPixmap->use_gpu_copy = save_use_gpu_copy; -    pPixmap->devKind = save_pitch; - -out: -    REGION_UNINIT(pScreen, &ValidReg); -    return ret; -} - -/** - * Performs migration of the pixmaps according to the operation information - * provided in pixmaps and can_accel and the migration scheme chosen in the - * config file. - */ -void -exaDoMigration_classic (ExaMigrationPtr pixmaps, int npixmaps, Bool can_accel) -{ -    ScreenPtr pScreen = pixmaps[0].pPix->drawable.pScreen; -    ExaScreenPriv(pScreen); -    int i, j; - -    /* If this debugging flag is set, check each pixmap for whether it is marked -     * as clean, and if so, actually check if that's the case.  This should help -     * catch issues with failing to mark a drawable as dirty.  While it will -     * catch them late (after the operation happened), it at least explains what -     * went wrong, and instrumenting the code to find what operation happened -     * to the pixmap last shouldn't be hard. -     */ -    if (pExaScr->checkDirtyCorrectness) { -	for (i = 0; i < npixmaps; i++) { -	    if (!exaPixmapIsDirty (pixmaps[i].pPix) && -		!exaAssertNotDirty (pixmaps[i].pPix)) -		ErrorF("%s: Pixmap %d dirty but not marked as such!\n", __func__, i); -	} -    } -    /* If anything is pinned in system memory, we won't be able to -     * accelerate. -     */ -    for (i = 0; i < npixmaps; i++) { -	if (exaPixmapIsPinned (pixmaps[i].pPix) && -	    !exaPixmapHasGpuCopy (pixmaps[i].pPix)) -	{ -	    EXA_FALLBACK(("Pixmap %p (%dx%d) pinned in sys\n", pixmaps[i].pPix, -		      pixmaps[i].pPix->drawable.width, -		      pixmaps[i].pPix->drawable.height)); -	    can_accel = FALSE; -	    break; -	} -    } - -    if (pExaScr->migration == ExaMigrationSmart) { -	/* If we've got something as a destination that we shouldn't cause to -	 * become newly dirtied, take the unaccelerated route. -	 */ -	for (i = 0; i < npixmaps; i++) { -	    if (pixmaps[i].as_dst && !exaPixmapShouldBeInFB (pixmaps[i].pPix) && -		!exaPixmapIsDirty (pixmaps[i].pPix)) -	    { -		for (i = 0; i < npixmaps; i++) { -		    if (!exaPixmapIsDirty (pixmaps[i].pPix)) -			exaDoMoveOutPixmap (pixmaps + i); -		} -		return; -	    } -	} - -	/* If we aren't going to accelerate, then we migrate everybody toward -	 * system memory, and kick out if it's free. -	 */ -	if (!can_accel) { -	    for (i = 0; i < npixmaps; i++) { -		exaMigrateTowardSys (pixmaps + i); -		if (!exaPixmapIsDirty (pixmaps[i].pPix)) -		    exaDoMoveOutPixmap (pixmaps + i); -	    } -	    return; -	} - -	/* Finally, the acceleration path.  Move them all in. */ -	for (i = 0; i < npixmaps; i++) { -	    exaMigrateTowardFb(pixmaps + i); -	    exaDoMoveInPixmap(pixmaps + i); -	} -    } else if (pExaScr->migration == ExaMigrationGreedy) { -	/* If we can't accelerate, either because the driver can't or because one of -	 * the pixmaps is pinned in system memory, then we migrate everybody toward -	 * system memory. -	 * -	 * We also migrate toward system if all pixmaps involved are currently in -	 * system memory -- this can mitigate thrashing when there are significantly -	 * more pixmaps active than would fit in memory. -	 * -	 * If not, then we migrate toward FB so that hopefully acceleration can -	 * happen. -	 */ -	if (!can_accel) { -	    for (i = 0; i < npixmaps; i++) -		exaMigrateTowardSys (pixmaps + i); -	    return; -	} - -	for (i = 0; i < npixmaps; i++) { -	    if (exaPixmapHasGpuCopy(pixmaps[i].pPix)) { -		/* Found one in FB, so move all to FB. */ -		for (j = 0; j < npixmaps; j++) -		    exaMigrateTowardFb(pixmaps + i); -		return; -	    } -	} - -	/* Nobody's in FB, so move all away from FB. */ -	for (i = 0; i < npixmaps; i++) -	    exaMigrateTowardSys(pixmaps + i); -    } else if (pExaScr->migration == ExaMigrationAlways) { -	/* Always move the pixmaps out if we can't accelerate.  If we can -	 * accelerate, try to move them all in.  If that fails, then move them -	 * back out. -	 */ -	if (!can_accel) { -	    for (i = 0; i < npixmaps; i++) -		exaDoMoveOutPixmap(pixmaps + i); -	    return; -	} - -	/* Now, try to move them all into FB */ -	for (i = 0; i < npixmaps; i++) { -	    exaDoMoveInPixmap(pixmaps + i); -	} - -	/* If we couldn't fit everything in, abort */ -	for (i = 0; i < npixmaps; i++) { -	    if (!exaPixmapHasGpuCopy(pixmaps[i].pPix)) { -		return; -	    } -	} - -	/* Yay, everything has a gpu copy, mark memory as used */ -	for (i = 0; i < npixmaps; i++) { -	    ExaOffscreenMarkUsed (pixmaps[i].pPix); -	} -    } -} - -void -exaPrepareAccessReg_classic(PixmapPtr pPixmap, int index, RegionPtr pReg) -{ -    ExaMigrationRec pixmaps[1]; - -    if (index == EXA_PREPARE_DEST || index == EXA_PREPARE_AUX_DEST) { -	pixmaps[0].as_dst = TRUE; -	pixmaps[0].as_src = FALSE; -    } else { -	pixmaps[0].as_dst = FALSE; -	pixmaps[0].as_src = TRUE; -    } -    pixmaps[0].pPix = pPixmap; -    pixmaps[0].pReg = pReg; - -    exaDoMigration(pixmaps, 1, FALSE); - -    (void)ExaDoPrepareAccess(pPixmap, index); -} +/*
 + * Copyright © 2006 Intel Corporation
 + *
 + * Permission is hereby granted, free of charge, to any person obtaining a
 + * copy of this software and associated documentation files (the "Software"),
 + * to deal in the Software without restriction, including without limitation
 + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
 + * and/or sell copies of the Software, and to permit persons to whom the
 + * Software is furnished to do so, subject to the following conditions:
 + *
 + * The above copyright notice and this permission notice (including the next
 + * paragraph) shall be included in all copies or substantial portions of the
 + * Software.
 + *
 + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
 + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 + * SOFTWARE.
 + *
 + * Authors:
 + *    Eric Anholt <eric@anholt.net>
 + *    Michel Dänzer <michel@tungstengraphics.com>
 + *
 + */
 +
 +#ifdef HAVE_DIX_CONFIG_H
 +#include <dix-config.h>
 +#endif
 +
 +#include <string.h>
 +
 +#include "exa_priv.h"
 +#include "exa.h"
 +
 +#if DEBUG_MIGRATE
 +#define DBG_MIGRATE(a) ErrorF a
 +#else
 +#define DBG_MIGRATE(a)
 +#endif
 +
 +/**
 + * The fallback path for UTS/DFS failing is to just memcpy.  exaCopyDirtyToSys
 + * and exaCopyDirtyToFb both needed to do this loop.
 + */
 +static void
 +exaMemcpyBox (PixmapPtr pPixmap, BoxPtr pbox, CARD8 *src, int src_pitch,
 +	      CARD8 *dst, int dst_pitch)
 + {
 +    int i, cpp = pPixmap->drawable.bitsPerPixel / 8;
 +    int bytes = (pbox->x2 - pbox->x1) * cpp;
 +
 +    src += pbox->y1 * src_pitch + pbox->x1 * cpp;
 +    dst += pbox->y1 * dst_pitch + pbox->x1 * cpp;
 +
 +    for (i = pbox->y2 - pbox->y1; i; i--) {
 +	memcpy (dst, src, bytes);
 +	src += src_pitch;
 +	dst += dst_pitch;
 +    }
 +}
 +
 +/**
 + * Returns TRUE if the pixmap is dirty (has been modified in its current
 + * location compared to the other), or lacks a private for tracking
 + * dirtiness.
 + */
 +static Bool
 +exaPixmapIsDirty (PixmapPtr pPix)
 +{
 +    ExaPixmapPriv (pPix);
 +
 +    if (pExaPixmap == NULL)
 +	EXA_FatalErrorDebugWithRet(("EXA bug: exaPixmapIsDirty was called on a non-exa pixmap.\n"), TRUE);
 +
 +    if (!pExaPixmap->pDamage)
 +	return FALSE;
 +
 +    return REGION_NOTEMPTY (pScreen, DamageRegion(pExaPixmap->pDamage)) ||
 +	!REGION_EQUAL(pScreen, &pExaPixmap->validSys, &pExaPixmap->validFB);
 +}
 +
 +/**
 + * Returns TRUE if the pixmap is either pinned in FB, or has a sufficient score
 + * to be considered "should be in framebuffer".  That's just anything that has
 + * had more acceleration than fallbacks, or has no score yet.
 + *
 + * Only valid if using a migration scheme that tracks score.
 + */
 +static Bool
 +exaPixmapShouldBeInFB (PixmapPtr pPix)
 +{
 +    ExaPixmapPriv (pPix);
 +
 +    if (exaPixmapIsPinned (pPix))
 +	return TRUE;
 +
 +    return pExaPixmap->score >= 0;
 +}
 +
 +/**
 + * If the pixmap is currently dirty, this copies at least the dirty area from
 + * FB to system or vice versa.  Both areas must be allocated.
 + */
 +static void
 +exaCopyDirty(ExaMigrationPtr migrate, RegionPtr pValidDst, RegionPtr pValidSrc,
 +	     Bool (*transfer) (PixmapPtr pPix, int x, int y, int w, int h,
 +			       char *sys, int sys_pitch), int fallback_index,
 +	     void (*sync) (ScreenPtr pScreen))
 +{
 +    PixmapPtr pPixmap = migrate->pPix;
 +    ExaPixmapPriv (pPixmap);
 +    RegionPtr damage = DamageRegion (pExaPixmap->pDamage);
 +    RegionRec CopyReg;
 +    Bool save_use_gpu_copy;
 +    int save_pitch;
 +    BoxPtr pBox;
 +    int nbox;
 +    Bool access_prepared = FALSE;
 +    Bool need_sync = FALSE;
 +
 +    /* Damaged bits are valid in current copy but invalid in other one */
 +    if (pExaPixmap->use_gpu_copy) {
 +	REGION_UNION(pScreen, &pExaPixmap->validFB, &pExaPixmap->validFB,
 +		     damage);
 +	REGION_SUBTRACT(pScreen, &pExaPixmap->validSys, &pExaPixmap->validSys,
 +			damage);
 +    } else {
 +	REGION_UNION(pScreen, &pExaPixmap->validSys, &pExaPixmap->validSys,
 +		     damage);
 +	REGION_SUBTRACT(pScreen, &pExaPixmap->validFB, &pExaPixmap->validFB,
 +			damage);
 +    }
 +
 +    REGION_EMPTY(pScreen, damage);
 +
 +    /* Copy bits valid in source but not in destination */
 +    REGION_NULL(pScreen, &CopyReg);
 +    REGION_SUBTRACT(pScreen, &CopyReg, pValidSrc, pValidDst);
 +
 +    if (migrate->as_dst) {
 +	ExaScreenPriv (pPixmap->drawable.pScreen);
 +
 +	/* XXX: The pending damage region will be marked as damaged after the
 +	 * operation, so it should serve as an upper bound for the region that
 +	 * needs to be synchronized for the operation. Unfortunately, this
 +	 * causes corruption in some cases, e.g. when starting compiz. See
 +	 * https://bugs.freedesktop.org/show_bug.cgi?id=12916 .
 +	 */
 +	if (pExaScr->optimize_migration) {
 +	    RegionPtr pending_damage = DamagePendingRegion(pExaPixmap->pDamage);
 +
 +#if DEBUG_MIGRATE
 +	    if (REGION_NIL(pending_damage)) {
 +		static Bool firsttime = TRUE;
 +
 +		if (firsttime) {
 +		    ErrorF("%s: Pending damage region empty!\n", __func__);
 +		    firsttime = FALSE;
 +		}
 +	    }
 +#endif
 +
 +	    /* Try to prevent destination valid region from growing too many
 +	     * rects by filling it up to the extents of the union of the
 +	     * destination valid region and the pending damage region.
 +	     */
 +	    if (REGION_NUM_RECTS(pValidDst) > 10) {
 +		BoxRec box;
 +		BoxPtr pValidExt, pDamageExt;
 +		RegionRec closure;
 +
 +		pValidExt = REGION_EXTENTS(pScreen, pValidDst);
 +		pDamageExt = REGION_EXTENTS(pScreen, pending_damage);
 +
 +		box.x1 = min(pValidExt->x1, pDamageExt->x1);
 +		box.y1 = min(pValidExt->y1, pDamageExt->y1);
 +		box.x2 = max(pValidExt->x2, pDamageExt->x2);
 +		box.y2 = max(pValidExt->y2, pDamageExt->y2);
 +
 +		REGION_INIT(pScreen, &closure, &box, 0);
 +		REGION_INTERSECT(pScreen, &CopyReg, &CopyReg, &closure);
 +	    } else
 +		REGION_INTERSECT(pScreen, &CopyReg, &CopyReg, pending_damage);
 +	}
 +
 +	/* The caller may provide a region to be subtracted from the calculated
 +	 * dirty region. This is to avoid migration of bits that don't
 +	 * contribute to the result of the operation.
 +	 */
 +	if (migrate->pReg)
 +	    REGION_SUBTRACT(pScreen, &CopyReg, &CopyReg, migrate->pReg);
 +    } else {
 +	/* The caller may restrict the region to be migrated for source pixmaps
 +	 * to what's relevant for the operation.
 +	 */
 +	if (migrate->pReg)
 +	    REGION_INTERSECT(pScreen, &CopyReg, &CopyReg, migrate->pReg);
 +    }
 +
 +    pBox = REGION_RECTS(&CopyReg);
 +    nbox = REGION_NUM_RECTS(&CopyReg);
 +
 +    save_use_gpu_copy = pExaPixmap->use_gpu_copy;
 +    save_pitch = pPixmap->devKind;
 +    pExaPixmap->use_gpu_copy = TRUE;
 +    pPixmap->devKind = pExaPixmap->fb_pitch;
 +
 +    while (nbox--) {
 +	pBox->x1 = max(pBox->x1, 0);
 +	pBox->y1 = max(pBox->y1, 0);
 +	pBox->x2 = min(pBox->x2, pPixmap->drawable.width);
 +	pBox->y2 = min(pBox->y2, pPixmap->drawable.height);
 +
 +	if (pBox->x1 >= pBox->x2 || pBox->y1 >= pBox->y2)
 +	    continue;
 +
 +	if (!transfer || !transfer (pPixmap,
 +				    pBox->x1, pBox->y1,
 +				    pBox->x2 - pBox->x1,
 +				    pBox->y2 - pBox->y1,
 +				    (char *) (pExaPixmap->sys_ptr
 +				    + pBox->y1 * pExaPixmap->sys_pitch
 +				    + pBox->x1 * pPixmap->drawable.bitsPerPixel / 8),
 +				    pExaPixmap->sys_pitch))
 +	{
 +	    if (!access_prepared) {
 +		ExaDoPrepareAccess(pPixmap, fallback_index);
 +		access_prepared = TRUE;
 +	    }
 +	    if (fallback_index == EXA_PREPARE_DEST) {
 +		exaMemcpyBox (pPixmap, pBox,
 +			      pExaPixmap->sys_ptr, pExaPixmap->sys_pitch,
 +			      pPixmap->devPrivate.ptr, pPixmap->devKind);
 +	    } else {
 +		exaMemcpyBox (pPixmap, pBox,
 +			      pPixmap->devPrivate.ptr, pPixmap->devKind,
 +			      pExaPixmap->sys_ptr, pExaPixmap->sys_pitch);
 +	    }
 +	} else
 +	    need_sync = TRUE;
 +
 +	pBox++;
 +    }
 +
 +    pExaPixmap->use_gpu_copy = save_use_gpu_copy;
 +    pPixmap->devKind = save_pitch;
 +
 +    /* Try to prevent source valid region from growing too many rects by
 +     * removing parts of it which are also in the destination valid region.
 +     * Removing anything beyond that would lead to data loss.
 +     */
 +    if (REGION_NUM_RECTS(pValidSrc) > 20)
 +	REGION_SUBTRACT(pScreen, pValidSrc, pValidSrc, pValidDst);
 +
 +    /* The copied bits are now valid in destination */
 +    REGION_UNION(pScreen, pValidDst, pValidDst, &CopyReg);
 +
 +    REGION_UNINIT(pScreen, &CopyReg);
 +
 +    if (access_prepared)
 +	exaFinishAccess(&pPixmap->drawable, fallback_index);
 +    else if (need_sync && sync)
 +	sync (pPixmap->drawable.pScreen);
 +}
 +
 +/**
 + * If the pixmap is currently dirty, this copies at least the dirty area from
 + * the framebuffer  memory copy to the system memory copy.  Both areas must be
 + * allocated.
 + */
 +void
 +exaCopyDirtyToSys (ExaMigrationPtr migrate)
 +{
 +    PixmapPtr pPixmap = migrate->pPix;
 +    ExaScreenPriv (pPixmap->drawable.pScreen);
 +    ExaPixmapPriv (pPixmap);
 +
 +    exaCopyDirty(migrate, &pExaPixmap->validSys, &pExaPixmap->validFB,
 +		 pExaScr->info->DownloadFromScreen, EXA_PREPARE_SRC,
 +		 exaWaitSync);
 +}
 +
 +/**
 + * If the pixmap is currently dirty, this copies at least the dirty area from
 + * the system memory copy to the framebuffer memory copy.  Both areas must be
 + * allocated.
 + */
 +void
 +exaCopyDirtyToFb (ExaMigrationPtr migrate)
 +{
 +    PixmapPtr pPixmap = migrate->pPix;
 +    ExaScreenPriv (pPixmap->drawable.pScreen);
 +    ExaPixmapPriv (pPixmap);
 +
 +    exaCopyDirty(migrate, &pExaPixmap->validFB, &pExaPixmap->validSys,
 +		 pExaScr->info->UploadToScreen, EXA_PREPARE_DEST, NULL);
 +}
 +
 +/**
 + * Allocates a framebuffer copy of the pixmap if necessary, and then copies
 + * any necessary pixmap data into the framebuffer copy and points the pixmap at
 + * it.
 + *
 + * Note that when first allocated, a pixmap will have FALSE dirty flag.
 + * This is intentional because pixmap data starts out undefined.  So if we move
 + * it in due to the first operation against it being accelerated, it will have
 + * undefined framebuffer contents that we didn't have to upload.  If we do
 + * moveouts (and moveins) after the first movein, then we will only have to copy
 + * back and forth if the pixmap was written to after the last synchronization of
 + * the two copies.  Then, at exaPixmapSave (when the framebuffer copy goes away)
 + * we mark the pixmap dirty, so that the next exaMoveInPixmap will actually move
 + * all the data, since it's almost surely all valid now.
 + */
 +static void
 +exaDoMoveInPixmap (ExaMigrationPtr migrate)
 +{
 +    PixmapPtr pPixmap = migrate->pPix;
 +    ScreenPtr pScreen = pPixmap->drawable.pScreen;
 +    ExaScreenPriv (pScreen);
 +    ExaPixmapPriv (pPixmap);
 +
 +    /* If we're VT-switched away, no touching card memory allowed. */
 +    if (pExaScr->swappedOut)
 +	return;
 +
 +    /* If we're not allowed to move, then fail. */
 +    if (exaPixmapIsPinned(pPixmap))
 +	return;
 +
 +    /* Don't migrate in pixmaps which are less than 8bpp.  This avoids a lot of
 +     * fragility in EXA, and <8bpp is probably not used enough any more to care
 +     * (at least, not in acceleratd paths).
 +     */
 +    if (pPixmap->drawable.bitsPerPixel < 8)
 +	return;
 +
 +    if (pExaPixmap->accel_blocked)
 +	return;
 +
 +    if (pExaPixmap->area == NULL) {
 +	pExaPixmap->area =
 +	    exaOffscreenAlloc (pScreen, pExaPixmap->fb_size,
 +			       pExaScr->info->pixmapOffsetAlign, FALSE,
 +                               exaPixmapSave, (pointer) pPixmap);
 +	if (pExaPixmap->area == NULL)
 +	    return;
 +
 +	pExaPixmap->fb_ptr = (CARD8 *) pExaScr->info->memoryBase +
 +				       pExaPixmap->area->offset;
 +    }
 +
 +    exaCopyDirtyToFb (migrate);
 +
 +    if (exaPixmapHasGpuCopy(pPixmap))
 +	return;
 +
 +    DBG_MIGRATE (("-> %p (0x%x) (%dx%d) (%c)\n", pPixmap,
 +		  (ExaGetPixmapPriv(pPixmap)->area ?
 +                   ExaGetPixmapPriv(pPixmap)->area->offset : 0),
 +		  pPixmap->drawable.width,
 +		  pPixmap->drawable.height,
 +		  exaPixmapIsDirty(pPixmap) ? 'd' : 'c'));
 +
 +    pExaPixmap->use_gpu_copy = TRUE;
 +
 +    pPixmap->devKind = pExaPixmap->fb_pitch;
 +    pPixmap->drawable.serialNumber = NEXT_SERIAL_NUMBER;
 +}
 +
 +void
 +exaMoveInPixmap_classic (PixmapPtr pPixmap)
 +{
 +    static ExaMigrationRec migrate = { FALSE, TRUE,
 +				       NULL, NULL };
 +
 +    migrate.pPix = pPixmap;
 +    exaDoMoveInPixmap (&migrate);
 +}
 +
 +/**
 + * Switches the current active location of the pixmap to system memory, copying
 + * updated data out if necessary.
 + */
 +static void
 +exaDoMoveOutPixmap (ExaMigrationPtr migrate)
 +{
 +    PixmapPtr pPixmap = migrate->pPix;
 +    ExaPixmapPriv (pPixmap);
 +
 +    if (!pExaPixmap->area || exaPixmapIsPinned(pPixmap))
 +	return;
 +
 +    exaCopyDirtyToSys (migrate);
 +
 +    if (exaPixmapHasGpuCopy(pPixmap)) {
 +
 +	DBG_MIGRATE (("<- %p (%p) (%dx%d) (%c)\n", pPixmap,
 +		      (void*)(ExaGetPixmapPriv(pPixmap)->area ?
 +			      ExaGetPixmapPriv(pPixmap)->area->offset : 0),
 +		      pPixmap->drawable.width,
 +		      pPixmap->drawable.height,
 +		      exaPixmapIsDirty(pPixmap) ? 'd' : 'c'));
 +
 +	pExaPixmap->use_gpu_copy = FALSE;
 +
 +	pPixmap->devKind = pExaPixmap->sys_pitch;
 +	pPixmap->drawable.serialNumber = NEXT_SERIAL_NUMBER;
 +    }
 +}
 +
 +void
 +exaMoveOutPixmap_classic (PixmapPtr pPixmap)
 +{
 +    static ExaMigrationRec migrate = { FALSE, TRUE,
 +				       NULL, NULL };
 +
 +    migrate.pPix = pPixmap;
 +    exaDoMoveOutPixmap (&migrate);
 +}
 +
 +
 +/**
 + * Copies out important pixmap data and removes references to framebuffer area.
 + * Called when the memory manager decides it's time to kick the pixmap out of
 + * framebuffer entirely.
 + */
 +void
 +exaPixmapSave (ScreenPtr pScreen, ExaOffscreenArea *area)
 +{
 +    PixmapPtr pPixmap = area->privData;
 +    ExaPixmapPriv(pPixmap);
 +
 +    exaMoveOutPixmap(pPixmap);
 +
 +    pExaPixmap->fb_ptr = NULL;
 +    pExaPixmap->area = NULL;
 +
 +    /* Mark all FB bits as invalid, so all valid system bits get copied to FB
 +     * next time */
 +    REGION_EMPTY(pPixmap->drawable.pScreen, &pExaPixmap->validFB);
 +}
 +
 +/**
 + * For the "greedy" migration scheme, pushes the pixmap toward being located in
 + * framebuffer memory.
 + */
 +static void
 +exaMigrateTowardFb (ExaMigrationPtr migrate)
 +{
 +    PixmapPtr pPixmap = migrate->pPix;
 +    ExaPixmapPriv (pPixmap);
 +
 +    if (pExaPixmap->score == EXA_PIXMAP_SCORE_PINNED) {
 +	DBG_MIGRATE(("UseScreen: not migrating pinned pixmap %p\n",
 +		     (pointer)pPixmap));
 +	return;
 +    }
 +
 +    DBG_MIGRATE(("UseScreen %p score %d\n",
 +		 (pointer)pPixmap, pExaPixmap->score));
 +
 +    if (pExaPixmap->score == EXA_PIXMAP_SCORE_INIT) {
 +	exaDoMoveInPixmap(migrate);
 +	pExaPixmap->score = 0;
 +    }
 +
 +    if (pExaPixmap->score < EXA_PIXMAP_SCORE_MAX)
 +	pExaPixmap->score++;
 +
 +    if (pExaPixmap->score >= EXA_PIXMAP_SCORE_MOVE_IN &&
 +	!exaPixmapHasGpuCopy(pPixmap))
 +    {
 +	exaDoMoveInPixmap(migrate);
 +    }
 +
 +    if (exaPixmapHasGpuCopy(pPixmap)) {
 +	exaCopyDirtyToFb (migrate);
 +	ExaOffscreenMarkUsed (pPixmap);
 +    } else
 +	exaCopyDirtyToSys (migrate);
 +}
 +
 +/**
 + * For the "greedy" migration scheme, pushes the pixmap toward being located in
 + * system memory.
 + */
 +static void
 +exaMigrateTowardSys (ExaMigrationPtr migrate)
 +{
 +    PixmapPtr pPixmap = migrate->pPix;
 +    ExaPixmapPriv (pPixmap);
 +
 +    DBG_MIGRATE(("UseMem: %p score %d\n", (pointer)pPixmap, pExaPixmap->score));
 +
 +    if (pExaPixmap->score == EXA_PIXMAP_SCORE_PINNED)
 +	return;
 +
 +    if (pExaPixmap->score == EXA_PIXMAP_SCORE_INIT)
 +	pExaPixmap->score = 0;
 +
 +    if (pExaPixmap->score > EXA_PIXMAP_SCORE_MIN)
 +	pExaPixmap->score--;
 +
 +    if (pExaPixmap->score <= EXA_PIXMAP_SCORE_MOVE_OUT && pExaPixmap->area)
 +	exaDoMoveOutPixmap(migrate);
 +
 +    if (exaPixmapHasGpuCopy(pPixmap)) {
 +	exaCopyDirtyToFb (migrate);
 +	ExaOffscreenMarkUsed (pPixmap);
 +    } else
 +	exaCopyDirtyToSys (migrate);
 +}
 +
 +/**
 + * If the pixmap has both a framebuffer and system memory copy, this function
 + * asserts that both of them are the same.
 + */
 +static Bool
 +exaAssertNotDirty (PixmapPtr pPixmap)
 +{
 +    ExaPixmapPriv (pPixmap);
 +    CARD8 *dst, *src;
 +    RegionRec ValidReg;
 +    int dst_pitch, src_pitch, cpp, y, nbox, save_pitch;
 +    BoxPtr pBox;
 +    Bool ret = TRUE, save_use_gpu_copy;
 +
 +    if (exaPixmapIsPinned(pPixmap) || pExaPixmap->area == NULL)
 +	return ret;
 +
 +    REGION_NULL(pScreen, &ValidReg);
 +    REGION_INTERSECT(pScreen, &ValidReg, &pExaPixmap->validFB,
 +		     &pExaPixmap->validSys);
 +    nbox = REGION_NUM_RECTS(&ValidReg);
 +
 +    if (!nbox)
 +	goto out;
 +
 +    pBox = REGION_RECTS(&ValidReg);
 +
 +    dst_pitch = pExaPixmap->sys_pitch;
 +    src_pitch = pExaPixmap->fb_pitch;
 +    cpp = pPixmap->drawable.bitsPerPixel / 8;
 +
 +    save_use_gpu_copy = pExaPixmap->use_gpu_copy;
 +    save_pitch = pPixmap->devKind;
 +    pExaPixmap->use_gpu_copy = TRUE;
 +    pPixmap->devKind = pExaPixmap->fb_pitch;
 +
 +    if (!ExaDoPrepareAccess(pPixmap, EXA_PREPARE_SRC))
 +	goto skip;
 +
 +    while (nbox--) {
 +	    int rowbytes;
 +
 +	    pBox->x1 = max(pBox->x1, 0);
 +	    pBox->y1 = max(pBox->y1, 0);
 +	    pBox->x2 = min(pBox->x2, pPixmap->drawable.width);
 +	    pBox->y2 = min(pBox->y2, pPixmap->drawable.height);
 +
 +	    if (pBox->x1 >= pBox->x2 || pBox->y1 >= pBox->y2)
 +		continue;
 +
 +	    rowbytes = (pBox->x2 - pBox->x1) * cpp;
 +	    src = (CARD8 *) pPixmap->devPrivate.ptr + pBox->y1 * src_pitch + pBox->x1 * cpp;
 +	    dst = pExaPixmap->sys_ptr + pBox->y1 * dst_pitch + pBox->x1 * cpp;
 +
 +	    for (y = pBox->y1; y < pBox->y2;
 +		 y++, src += src_pitch, dst += dst_pitch) {
 +		if (memcmp(dst, src, rowbytes) != 0) {
 +		    ret = FALSE;
 +		    exaPixmapDirty(pPixmap, pBox->x1, pBox->y1, pBox->x2,
 +				   pBox->y2);
 +		    break;
 +		}
 +	    }
 +    }
 +
 +skip:
 +    exaFinishAccess(&pPixmap->drawable, EXA_PREPARE_SRC);
 +
 +    pExaPixmap->use_gpu_copy = save_use_gpu_copy;
 +    pPixmap->devKind = save_pitch;
 +
 +out:
 +    REGION_UNINIT(pScreen, &ValidReg);
 +    return ret;
 +}
 +
 +/**
 + * Performs migration of the pixmaps according to the operation information
 + * provided in pixmaps and can_accel and the migration scheme chosen in the
 + * config file.
 + */
 +void
 +exaDoMigration_classic (ExaMigrationPtr pixmaps, int npixmaps, Bool can_accel)
 +{
 +    ScreenPtr pScreen = pixmaps[0].pPix->drawable.pScreen;
 +    ExaScreenPriv(pScreen);
 +    int i, j;
 +
 +    /* If this debugging flag is set, check each pixmap for whether it is marked
 +     * as clean, and if so, actually check if that's the case.  This should help
 +     * catch issues with failing to mark a drawable as dirty.  While it will
 +     * catch them late (after the operation happened), it at least explains what
 +     * went wrong, and instrumenting the code to find what operation happened
 +     * to the pixmap last shouldn't be hard.
 +     */
 +    if (pExaScr->checkDirtyCorrectness) {
 +	for (i = 0; i < npixmaps; i++) {
 +	    if (!exaPixmapIsDirty (pixmaps[i].pPix) &&
 +		!exaAssertNotDirty (pixmaps[i].pPix))
 +		ErrorF("%s: Pixmap %d dirty but not marked as such!\n", __FUNCTION__, i);
 +	}
 +    }
 +    /* If anything is pinned in system memory, we won't be able to
 +     * accelerate.
 +     */
 +    for (i = 0; i < npixmaps; i++) {
 +	if (exaPixmapIsPinned (pixmaps[i].pPix) &&
 +	    !exaPixmapHasGpuCopy (pixmaps[i].pPix))
 +	{
 +	    EXA_FALLBACK(("Pixmap %p (%dx%d) pinned in sys\n", pixmaps[i].pPix,
 +		      pixmaps[i].pPix->drawable.width,
 +		      pixmaps[i].pPix->drawable.height));
 +	    can_accel = FALSE;
 +	    break;
 +	}
 +    }
 +
 +    if (pExaScr->migration == ExaMigrationSmart) {
 +	/* If we've got something as a destination that we shouldn't cause to
 +	 * become newly dirtied, take the unaccelerated route.
 +	 */
 +	for (i = 0; i < npixmaps; i++) {
 +	    if (pixmaps[i].as_dst && !exaPixmapShouldBeInFB (pixmaps[i].pPix) &&
 +		!exaPixmapIsDirty (pixmaps[i].pPix))
 +	    {
 +		for (i = 0; i < npixmaps; i++) {
 +		    if (!exaPixmapIsDirty (pixmaps[i].pPix))
 +			exaDoMoveOutPixmap (pixmaps + i);
 +		}
 +		return;
 +	    }
 +	}
 +
 +	/* If we aren't going to accelerate, then we migrate everybody toward
 +	 * system memory, and kick out if it's free.
 +	 */
 +	if (!can_accel) {
 +	    for (i = 0; i < npixmaps; i++) {
 +		exaMigrateTowardSys (pixmaps + i);
 +		if (!exaPixmapIsDirty (pixmaps[i].pPix))
 +		    exaDoMoveOutPixmap (pixmaps + i);
 +	    }
 +	    return;
 +	}
 +
 +	/* Finally, the acceleration path.  Move them all in. */
 +	for (i = 0; i < npixmaps; i++) {
 +	    exaMigrateTowardFb(pixmaps + i);
 +	    exaDoMoveInPixmap(pixmaps + i);
 +	}
 +    } else if (pExaScr->migration == ExaMigrationGreedy) {
 +	/* If we can't accelerate, either because the driver can't or because one of
 +	 * the pixmaps is pinned in system memory, then we migrate everybody toward
 +	 * system memory.
 +	 *
 +	 * We also migrate toward system if all pixmaps involved are currently in
 +	 * system memory -- this can mitigate thrashing when there are significantly
 +	 * more pixmaps active than would fit in memory.
 +	 *
 +	 * If not, then we migrate toward FB so that hopefully acceleration can
 +	 * happen.
 +	 */
 +	if (!can_accel) {
 +	    for (i = 0; i < npixmaps; i++)
 +		exaMigrateTowardSys (pixmaps + i);
 +	    return;
 +	}
 +
 +	for (i = 0; i < npixmaps; i++) {
 +	    if (exaPixmapHasGpuCopy(pixmaps[i].pPix)) {
 +		/* Found one in FB, so move all to FB. */
 +		for (j = 0; j < npixmaps; j++)
 +		    exaMigrateTowardFb(pixmaps + i);
 +		return;
 +	    }
 +	}
 +
 +	/* Nobody's in FB, so move all away from FB. */
 +	for (i = 0; i < npixmaps; i++)
 +	    exaMigrateTowardSys(pixmaps + i);
 +    } else if (pExaScr->migration == ExaMigrationAlways) {
 +	/* Always move the pixmaps out if we can't accelerate.  If we can
 +	 * accelerate, try to move them all in.  If that fails, then move them
 +	 * back out.
 +	 */
 +	if (!can_accel) {
 +	    for (i = 0; i < npixmaps; i++)
 +		exaDoMoveOutPixmap(pixmaps + i);
 +	    return;
 +	}
 +
 +	/* Now, try to move them all into FB */
 +	for (i = 0; i < npixmaps; i++) {
 +	    exaDoMoveInPixmap(pixmaps + i);
 +	}
 +
 +	/* If we couldn't fit everything in, abort */
 +	for (i = 0; i < npixmaps; i++) {
 +	    if (!exaPixmapHasGpuCopy(pixmaps[i].pPix)) {
 +		return;
 +	    }
 +	}
 +
 +	/* Yay, everything has a gpu copy, mark memory as used */
 +	for (i = 0; i < npixmaps; i++) {
 +	    ExaOffscreenMarkUsed (pixmaps[i].pPix);
 +	}
 +    }
 +}
 +
 +void
 +exaPrepareAccessReg_classic(PixmapPtr pPixmap, int index, RegionPtr pReg)
 +{
 +    ExaMigrationRec pixmaps[1];
 +
 +    if (index == EXA_PREPARE_DEST || index == EXA_PREPARE_AUX_DEST) {
 +	pixmaps[0].as_dst = TRUE;
 +	pixmaps[0].as_src = FALSE;
 +    } else {
 +	pixmaps[0].as_dst = FALSE;
 +	pixmaps[0].as_src = TRUE;
 +    }
 +    pixmaps[0].pPix = pPixmap;
 +    pixmaps[0].pReg = pReg;
 +
 +    exaDoMigration(pixmaps, 1, FALSE);
 +
 +    (void)ExaDoPrepareAccess(pPixmap, index);
 +}
 diff --git a/xorg-server/exa/exa_unaccel.c b/xorg-server/exa/exa_unaccel.c index b4ead7ff2..adbee8413 100644 --- a/xorg-server/exa/exa_unaccel.c +++ b/xorg-server/exa/exa_unaccel.c @@ -1,748 +1,748 @@ -/* - * - * Copyright © 1999 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. - */ - -#include "exa_priv.h" - -#ifdef RENDER -#include "mipict.h" -#endif - -/* - * These functions wrap the low-level fb rendering functions and - * synchronize framebuffer/accelerated drawing by stalling until - * the accelerator is idle - */ - -/** - * Calls exaPrepareAccess with EXA_PREPARE_SRC for the tile, if that is the - * current fill style. - * - * Solid doesn't use an extra pixmap source, and Stippled/OpaqueStippled are - * 1bpp and never in fb, so we don't worry about them. - * We should worry about them for completeness sake and going forward. - */ -void -exaPrepareAccessGC(GCPtr pGC) -{ -    if (pGC->stipple) -        exaPrepareAccess(&pGC->stipple->drawable, EXA_PREPARE_MASK); -    if (pGC->fillStyle == FillTiled) -	exaPrepareAccess(&pGC->tile.pixmap->drawable, EXA_PREPARE_SRC); -} - -/** - * Finishes access to the tile in the GC, if used. - */ -void -exaFinishAccessGC(GCPtr pGC) -{ -    if (pGC->fillStyle == FillTiled) -	exaFinishAccess(&pGC->tile.pixmap->drawable, EXA_PREPARE_SRC); -    if (pGC->stipple) -        exaFinishAccess(&pGC->stipple->drawable, EXA_PREPARE_MASK); -} - -#if DEBUG_TRACE_FALL -char -exaDrawableLocation(DrawablePtr pDrawable) -{ -    return exaDrawableIsOffscreen(pDrawable) ? 's' : 'm'; -} -#endif /* DEBUG_TRACE_FALL */ - -void -ExaCheckFillSpans (DrawablePtr pDrawable, GCPtr pGC, int nspans, -		   DDXPointPtr ppt, int *pwidth, int fSorted) -{ -    EXA_PRE_FALLBACK_GC(pGC); -    EXA_FALLBACK(("to %p (%c)\n", pDrawable, exaDrawableLocation(pDrawable))); -    exaPrepareAccess (pDrawable, EXA_PREPARE_DEST); -    exaPrepareAccessGC (pGC); -    pGC->ops->FillSpans (pDrawable, pGC, nspans, ppt, pwidth, fSorted); -    exaFinishAccessGC (pGC); -    exaFinishAccess (pDrawable, EXA_PREPARE_DEST); -    EXA_POST_FALLBACK_GC(pGC); -} - -void -ExaCheckSetSpans (DrawablePtr pDrawable, GCPtr pGC, char *psrc, -		 DDXPointPtr ppt, int *pwidth, int nspans, int fSorted) -{ -    EXA_PRE_FALLBACK_GC(pGC); -    EXA_FALLBACK(("to %p (%c)\n", pDrawable, exaDrawableLocation(pDrawable))); -    exaPrepareAccess (pDrawable, EXA_PREPARE_DEST); -    pGC->ops->SetSpans (pDrawable, pGC, psrc, ppt, pwidth, nspans, fSorted); -    exaFinishAccess (pDrawable, EXA_PREPARE_DEST); -    EXA_POST_FALLBACK_GC(pGC); -} - -void -ExaCheckPutImage (DrawablePtr pDrawable, GCPtr pGC, int depth, -		 int x, int y, int w, int h, int leftPad, int format, -		 char *bits) -{ -    PixmapPtr pPixmap = exaGetDrawablePixmap(pDrawable); -    ExaPixmapPriv(pPixmap); - -    EXA_PRE_FALLBACK_GC(pGC); -    EXA_FALLBACK(("to %p (%c)\n", pDrawable, exaDrawableLocation(pDrawable))); -    if (!pExaScr->prepare_access_reg || !pExaPixmap->pDamage || -	exaGCReadsDestination(pDrawable, pGC->planemask, pGC->fillStyle, -			      pGC->alu, pGC->clientClipType)) -	exaPrepareAccess (pDrawable, EXA_PREPARE_DEST); -    else -	pExaScr->prepare_access_reg(pPixmap, EXA_PREPARE_DEST, -				    DamagePendingRegion(pExaPixmap->pDamage)); -    pGC->ops->PutImage (pDrawable, pGC, depth, x, y, w, h, leftPad, format, bits); -    exaFinishAccess (pDrawable, EXA_PREPARE_DEST); -    EXA_POST_FALLBACK_GC(pGC); -} - -void -ExaCheckCopyNtoN (DrawablePtr pSrc, DrawablePtr pDst,  GCPtr pGC, -	     BoxPtr	pbox, int nbox, int dx, int dy, Bool	reverse,  -	     Bool upsidedown, Pixel bitplane, void *closure) -{ -    RegionRec reg; -    int xoff, yoff; -    EXA_PRE_FALLBACK_GC(pGC); -    EXA_FALLBACK(("from %p to %p (%c,%c)\n", pSrc, pDst, -		  exaDrawableLocation(pSrc), exaDrawableLocation(pDst))); - -    if (pExaScr->prepare_access_reg) { -	PixmapPtr pPixmap = exaGetDrawablePixmap(pSrc); - -	exaGetDrawableDeltas(pSrc, pPixmap, &xoff, &yoff); -	REGION_INIT(pScreen, ®, pbox, nbox); -	REGION_TRANSLATE(pScreen, ®, xoff + dx, yoff + dy); -	pExaScr->prepare_access_reg(pPixmap, EXA_PREPARE_SRC, ®); -	REGION_UNINIT(pScreen, ®); -    } else -	exaPrepareAccess (pSrc, EXA_PREPARE_SRC); - -    if (pExaScr->prepare_access_reg && -	!exaGCReadsDestination(pDst, pGC->planemask, pGC->fillStyle, -			       pGC->alu, pGC->clientClipType)) { -	PixmapPtr pPixmap = exaGetDrawablePixmap(pDst); - -	exaGetDrawableDeltas(pSrc, pPixmap, &xoff, &yoff); -	REGION_INIT(pScreen, ®, pbox, nbox); -	REGION_TRANSLATE(pScreen, ®, xoff, yoff); -	pExaScr->prepare_access_reg(pPixmap, EXA_PREPARE_DEST, ®); -	REGION_UNINIT(pScreen, ®); -    } else -	exaPrepareAccess (pDst, EXA_PREPARE_DEST); - -    /* This will eventually call fbCopyNtoN, with some calculation overhead. */ -    while (nbox--) { -	pGC->ops->CopyArea (pSrc, pDst, pGC, pbox->x1 - pSrc->x + dx, pbox->y1 - pSrc->y + dy,  -			pbox->x2 - pbox->x1, pbox->y2 - pbox->y1, pbox->x1 - pDst->x, pbox->y1 - pDst->y); -	pbox++; -    } -    exaFinishAccess (pSrc, EXA_PREPARE_SRC); -    exaFinishAccess (pDst, EXA_PREPARE_DEST); -    EXA_POST_FALLBACK_GC(pGC); -} - -static void -ExaFallbackPrepareReg(DrawablePtr pDrawable, -		      GCPtr pGC, -		      int x, int y, int width, int height, -		      int index, Bool checkReads) -{ -    ScreenPtr pScreen = pDrawable->pScreen; -    ExaScreenPriv(pScreen); - -    if (pExaScr->prepare_access_reg && -	!(checkReads && exaGCReadsDestination(pDrawable, -					      pGC->planemask, -					      pGC->fillStyle, -					      pGC->alu, -					      pGC->clientClipType))) { -	BoxRec box; -	RegionRec reg; -	int xoff, yoff; -	PixmapPtr pPixmap = exaGetDrawablePixmap(pDrawable); - -	exaGetDrawableDeltas(pDrawable, pPixmap, &xoff, &yoff); -	box.x1 = pDrawable->x + x + xoff; -	box.y1 = pDrawable->y + y + yoff; -	box.x2 = box.x1 + width; -	box.y2 = box.y1 + height; - -	REGION_INIT(pScreen, ®, &box, 1); -	pExaScr->prepare_access_reg(pPixmap, index, ®); -	REGION_UNINIT(pScreen, ®); -    } else -	exaPrepareAccess(pDrawable, index); -} - - -RegionPtr -ExaCheckCopyArea (DrawablePtr pSrc, DrawablePtr pDst, GCPtr pGC, -		 int srcx, int srcy, int w, int h, int dstx, int dsty) -{ -    RegionPtr ret; - -    EXA_PRE_FALLBACK_GC(pGC); -    EXA_FALLBACK(("from %p to %p (%c,%c)\n", pSrc, pDst, -		  exaDrawableLocation(pSrc), exaDrawableLocation(pDst))); -    ExaFallbackPrepareReg(pSrc, pGC, srcx, srcy, w, h, -			  EXA_PREPARE_SRC, FALSE); -    ExaFallbackPrepareReg(pDst, pGC, dstx, dsty, w, h, -			  EXA_PREPARE_DEST, TRUE); -    ret = pGC->ops->CopyArea (pSrc, pDst, pGC, srcx, srcy, w, h, dstx, dsty); -    exaFinishAccess (pSrc, EXA_PREPARE_SRC); -    exaFinishAccess (pDst, EXA_PREPARE_DEST); -    EXA_POST_FALLBACK_GC(pGC); - -    return ret; -} - -RegionPtr -ExaCheckCopyPlane (DrawablePtr pSrc, DrawablePtr pDst, GCPtr pGC, -		  int srcx, int srcy, int w, int h, int dstx, int dsty, -		  unsigned long bitPlane) -{ -    RegionPtr ret; - -    EXA_PRE_FALLBACK_GC(pGC); -    EXA_FALLBACK(("from %p to %p (%c,%c)\n", pSrc, pDst, -		  exaDrawableLocation(pSrc), exaDrawableLocation(pDst))); -    ExaFallbackPrepareReg(pSrc, pGC, srcx, srcy, w, h, -			  EXA_PREPARE_SRC, FALSE); -    ExaFallbackPrepareReg(pDst, pGC, dstx, dsty, w, h, -			  EXA_PREPARE_DEST, TRUE); -    ret = pGC->ops->CopyPlane (pSrc, pDst, pGC, srcx, srcy, w, h, dstx, dsty, -		       bitPlane); -    exaFinishAccess (pSrc, EXA_PREPARE_SRC); -    exaFinishAccess (pDst, EXA_PREPARE_DEST); -    EXA_POST_FALLBACK_GC(pGC); - -    return ret; -} - -void -ExaCheckPolyPoint (DrawablePtr pDrawable, GCPtr pGC, int mode, int npt, -		  DDXPointPtr pptInit) -{ -    EXA_PRE_FALLBACK_GC(pGC); -    EXA_FALLBACK(("to %p (%c)\n", pDrawable, exaDrawableLocation(pDrawable))); -    exaPrepareAccess (pDrawable, EXA_PREPARE_DEST); -    pGC->ops->PolyPoint (pDrawable, pGC, mode, npt, pptInit); -    exaFinishAccess (pDrawable, EXA_PREPARE_DEST); -    EXA_POST_FALLBACK_GC(pGC); -} - -void -ExaCheckPolylines (DrawablePtr pDrawable, GCPtr pGC, -		  int mode, int npt, DDXPointPtr ppt) -{ -    EXA_PRE_FALLBACK_GC(pGC); -    EXA_FALLBACK(("to %p (%c), width %d, mode %d, count %d\n", -		  pDrawable, exaDrawableLocation(pDrawable), -		  pGC->lineWidth, mode, npt)); - -    exaPrepareAccess (pDrawable, EXA_PREPARE_DEST); -    exaPrepareAccessGC (pGC); -    pGC->ops->Polylines (pDrawable, pGC, mode, npt, ppt); -    exaFinishAccessGC (pGC); -    exaFinishAccess (pDrawable, EXA_PREPARE_DEST); -    EXA_POST_FALLBACK_GC(pGC); -} - -void -ExaCheckPolySegment (DrawablePtr pDrawable, GCPtr pGC, -		    int nsegInit, xSegment *pSegInit) -{ -    EXA_PRE_FALLBACK_GC(pGC); -    EXA_FALLBACK(("to %p (%c) width %d, count %d\n", pDrawable, -		  exaDrawableLocation(pDrawable), pGC->lineWidth, nsegInit)); - -    exaPrepareAccess (pDrawable, EXA_PREPARE_DEST); -    exaPrepareAccessGC (pGC); -    pGC->ops->PolySegment (pDrawable, pGC, nsegInit, pSegInit); -    exaFinishAccessGC (pGC); -    exaFinishAccess (pDrawable, EXA_PREPARE_DEST); -    EXA_POST_FALLBACK_GC(pGC); -} - -void -ExaCheckPolyArc (DrawablePtr pDrawable, GCPtr pGC, -		int narcs, xArc *pArcs) -{ -    EXA_PRE_FALLBACK_GC(pGC); -    EXA_FALLBACK(("to %p (%c)\n", pDrawable, exaDrawableLocation(pDrawable))); - -    exaPrepareAccess (pDrawable, EXA_PREPARE_DEST); -    exaPrepareAccessGC (pGC); -    pGC->ops->PolyArc (pDrawable, pGC, narcs, pArcs); -    exaFinishAccessGC (pGC); -    exaFinishAccess (pDrawable, EXA_PREPARE_DEST); -    EXA_POST_FALLBACK_GC(pGC); -} - -void -ExaCheckPolyFillRect (DrawablePtr pDrawable, GCPtr pGC, -		     int nrect, xRectangle *prect) -{ -    EXA_PRE_FALLBACK_GC(pGC); -    EXA_FALLBACK(("to %p (%c)\n", pDrawable, exaDrawableLocation(pDrawable))); - -    exaPrepareAccess (pDrawable, EXA_PREPARE_DEST); -    exaPrepareAccessGC (pGC); -    pGC->ops->PolyFillRect (pDrawable, pGC, nrect, prect); -    exaFinishAccessGC (pGC); -    exaFinishAccess (pDrawable, EXA_PREPARE_DEST); -    EXA_POST_FALLBACK_GC(pGC); -} - -void -ExaCheckImageGlyphBlt (DrawablePtr pDrawable, GCPtr pGC, -		      int x, int y, unsigned int nglyph, -		      CharInfoPtr *ppci, pointer pglyphBase) -{ -    EXA_PRE_FALLBACK_GC(pGC); -    EXA_FALLBACK(("to %p (%c)\n", pDrawable, -		  exaDrawableLocation(pDrawable))); -    exaPrepareAccess (pDrawable, EXA_PREPARE_DEST); -    exaPrepareAccessGC (pGC); -    pGC->ops->ImageGlyphBlt (pDrawable, pGC, x, y, nglyph, ppci, pglyphBase); -    exaFinishAccessGC (pGC); -    exaFinishAccess (pDrawable, EXA_PREPARE_DEST); -    EXA_POST_FALLBACK_GC(pGC); -} - -void -ExaCheckPolyGlyphBlt (DrawablePtr pDrawable, GCPtr pGC, -		     int x, int y, unsigned int nglyph, -		     CharInfoPtr *ppci, pointer pglyphBase) -{ -    EXA_PRE_FALLBACK_GC(pGC); -    EXA_FALLBACK(("to %p (%c), style %d alu %d\n", pDrawable, -		  exaDrawableLocation(pDrawable), pGC->fillStyle, pGC->alu)); -    exaPrepareAccess (pDrawable, EXA_PREPARE_DEST); -    exaPrepareAccessGC (pGC); -    pGC->ops->PolyGlyphBlt (pDrawable, pGC, x, y, nglyph, ppci, pglyphBase); -    exaFinishAccessGC (pGC); -    exaFinishAccess (pDrawable, EXA_PREPARE_DEST); -    EXA_POST_FALLBACK_GC(pGC); -} - -void -ExaCheckPushPixels (GCPtr pGC, PixmapPtr pBitmap, -		   DrawablePtr pDrawable, -		   int w, int h, int x, int y) -{ -    EXA_PRE_FALLBACK_GC(pGC); -    EXA_FALLBACK(("from %p to %p (%c,%c)\n", pBitmap, pDrawable, -		  exaDrawableLocation(&pBitmap->drawable), -		  exaDrawableLocation(pDrawable))); -    ExaFallbackPrepareReg(pDrawable, pGC, x, y, w, h, -			  EXA_PREPARE_DEST, TRUE); -    ExaFallbackPrepareReg(&pBitmap->drawable, pGC, 0, 0, w, h, -			  EXA_PREPARE_SRC, FALSE); -    exaPrepareAccessGC (pGC); -    pGC->ops->PushPixels (pGC, pBitmap, pDrawable, w, h, x, y); -    exaFinishAccessGC (pGC); -    exaFinishAccess (&pBitmap->drawable, EXA_PREPARE_SRC); -    exaFinishAccess (pDrawable, EXA_PREPARE_DEST); -    EXA_POST_FALLBACK_GC(pGC); -} - -void -ExaCheckCopyWindow(WindowPtr pWin, DDXPointRec ptOldOrg, RegionPtr prgnSrc) -{ -    DrawablePtr pDrawable = &pWin->drawable; -    ScreenPtr pScreen = pDrawable->pScreen; -    EXA_PRE_FALLBACK(pScreen); -    EXA_FALLBACK(("from %p\n", pWin)); - -    /* Only need the source bits, the destination region will be overwritten */ -    if (pExaScr->prepare_access_reg) { -	PixmapPtr pPixmap = pScreen->GetWindowPixmap(pWin); -	int xoff, yoff; - -	exaGetDrawableDeltas(&pWin->drawable, pPixmap, &xoff, &yoff); -	REGION_TRANSLATE(pScreen, prgnSrc, xoff, yoff); -	pExaScr->prepare_access_reg(pPixmap, EXA_PREPARE_SRC, prgnSrc); -	REGION_TRANSLATE(pScreen, prgnSrc, -xoff, -yoff); -    } else -	exaPrepareAccess(pDrawable, EXA_PREPARE_SRC); - -    swap(pExaScr, pScreen, CopyWindow); -    pScreen->CopyWindow (pWin, ptOldOrg, prgnSrc); -    swap(pExaScr, pScreen, CopyWindow); -    exaFinishAccess (pDrawable, EXA_PREPARE_SRC); -    EXA_POST_FALLBACK(pScreen); -} - -void -ExaCheckGetImage(DrawablePtr pDrawable, int x, int y, int w, int h, -		unsigned int format, unsigned long planeMask, char *d) -{ -    ScreenPtr pScreen = pDrawable->pScreen; -    EXA_PRE_FALLBACK(pScreen); -    EXA_FALLBACK(("from %p (%c)\n", pDrawable, -		  exaDrawableLocation(pDrawable))); - -    ExaFallbackPrepareReg(pDrawable, NULL, x, y, w, h, -			  EXA_PREPARE_SRC, FALSE); -    swap(pExaScr, pScreen, GetImage); -    pScreen->GetImage (pDrawable, x, y, w, h, format, planeMask, d); -    swap(pExaScr, pScreen, GetImage); -    exaFinishAccess (pDrawable, EXA_PREPARE_SRC); -    EXA_POST_FALLBACK(pScreen); -} - -void -ExaCheckGetSpans (DrawablePtr pDrawable, -		 int wMax, -		 DDXPointPtr ppt, -		 int *pwidth, -		 int nspans, -		 char *pdstStart) -{ -    ScreenPtr pScreen = pDrawable->pScreen; - -    EXA_PRE_FALLBACK(pScreen); -    EXA_FALLBACK(("from %p (%c)\n", pDrawable, exaDrawableLocation(pDrawable))); -    exaPrepareAccess (pDrawable, EXA_PREPARE_SRC); -    swap(pExaScr, pScreen, GetSpans); -    pScreen->GetSpans (pDrawable, wMax, ppt, pwidth, nspans, pdstStart); -    swap(pExaScr, pScreen, GetSpans); -    exaFinishAccess (pDrawable, EXA_PREPARE_SRC); -    EXA_POST_FALLBACK(pScreen); -} - -static void -ExaSrcValidate(DrawablePtr pDrawable, -	       int x, -	       int y, -	       int width, -	       int height) -{ -    ScreenPtr pScreen = pDrawable->pScreen; -    ExaScreenPriv(pScreen); -    PixmapPtr pPix = exaGetDrawablePixmap (pDrawable); -    BoxRec box; -    RegionRec reg; -    RegionPtr dst; -    int xoff, yoff; - -    exaGetDrawableDeltas(pDrawable, pPix, &xoff, &yoff); - -    box.x1 = x + xoff; -    box.y1 = y + yoff; -    box.x2 = box.x1 + width; -    box.y2 = box.y1 + height; - -    dst = (pExaScr->srcPix == pPix) ? &pExaScr->srcReg : -	&pExaScr->maskReg; - -    REGION_INIT(pScreen, ®, &box, 1); -    REGION_UNION(pScreen, dst, dst, ®); -    REGION_UNINIT(pScreen, ®); - -    swap(pExaScr, pScreen, SourceValidate); -    pScreen->SourceValidate(pDrawable, x, y, width, height); -    swap(pExaScr, pScreen, SourceValidate); -} - -static Bool -ExaPrepareCompositeReg(ScreenPtr  pScreen, -		       CARD8      op, -		       PicturePtr pSrc, -		       PicturePtr pMask, -		       PicturePtr pDst, -		       INT16      xSrc, -		       INT16      ySrc, -		       INT16      xMask, -		       INT16      yMask, -		       INT16      xDst, -		       INT16      yDst, -		       CARD16     width, -		       CARD16     height) -{ -    RegionRec region; -    RegionPtr dstReg = NULL; -    RegionPtr srcReg = NULL; -    RegionPtr maskReg = NULL; -    PixmapPtr pSrcPix = NULL; -    PixmapPtr pMaskPix = NULL; -    PixmapPtr pDstPix; -    ExaScreenPriv(pScreen); -    Bool ret; - - -    REGION_NULL(pScreen, ®ion); - -    if (pSrc->pDrawable) { -	pSrcPix = exaGetDrawablePixmap(pSrc->pDrawable); -	REGION_NULL(pScreen, &pExaScr->srcReg); -	srcReg = &pExaScr->srcReg; -	pExaScr->srcPix = pSrcPix; -	if (pSrc != pDst) -	    REGION_TRANSLATE(pScreen, pSrc->pCompositeClip, -			     -pSrc->pDrawable->x, -			     -pSrc->pDrawable->y); -    } - -    if (pMask && pMask->pDrawable) { -	pMaskPix = exaGetDrawablePixmap(pMask->pDrawable); -	REGION_NULL(pScreen, &pExaScr->maskReg); -	maskReg = &pExaScr->maskReg; -	if (pMask != pDst && pMask != pSrc) -	    REGION_TRANSLATE(pScreen, pMask->pCompositeClip, -			     -pMask->pDrawable->x, -			     -pMask->pDrawable->y); -    } - -    REGION_TRANSLATE(pScreen, pDst->pCompositeClip, -		     -pDst->pDrawable->x, -		     -pDst->pDrawable->y); - -    pExaScr->SavedSourceValidate = ExaSrcValidate; -    swap(pExaScr, pScreen, SourceValidate); -    ret = miComputeCompositeRegion (®ion, pSrc, pMask, pDst, -				    xSrc, ySrc, xMask, yMask, -				    xDst, -				    yDst, -				    width, height); -    swap(pExaScr, pScreen, SourceValidate); - -    REGION_TRANSLATE(pScreen, pDst->pCompositeClip, -		     pDst->pDrawable->x, -		     pDst->pDrawable->y); -    if (pSrc->pDrawable && pSrc != pDst) -	REGION_TRANSLATE(pScreen, pSrc->pCompositeClip, -			 pSrc->pDrawable->x, -			 pSrc->pDrawable->y); -    if (pMask && pMask->pDrawable && pMask != pDst && pMask != pSrc) -	REGION_TRANSLATE(pScreen, pMask->pCompositeClip, -			 pMask->pDrawable->x, -			 pMask->pDrawable->y); - -    if (!ret) { -	if (srcReg) -	    REGION_UNINIT(pScreen, srcReg); -	if (maskReg) -	    REGION_UNINIT(pScreen, maskReg); - -	return FALSE; -    } - -    /** -     * Don't limit alphamaps readbacks for now until we've figured out how that -     * should be done. -     */ - -    if (pSrc->alphaMap && pSrc->alphaMap->pDrawable) -	pExaScr->prepare_access_reg(exaGetDrawablePixmap(pSrc->alphaMap->pDrawable), -				    EXA_PREPARE_AUX_SRC, -				    NULL); -    if (pMask && pMask->alphaMap && pMask->alphaMap->pDrawable) -	pExaScr->prepare_access_reg(exaGetDrawablePixmap(pMask->alphaMap->pDrawable), -				    EXA_PREPARE_AUX_MASK, -				    NULL); - -    if (pSrcPix) -	pExaScr->prepare_access_reg(pSrcPix, -				    EXA_PREPARE_SRC, -				    srcReg); - -    if (pMaskPix) -	pExaScr->prepare_access_reg(pMaskPix, -				    EXA_PREPARE_MASK, -				    maskReg); - -    if (srcReg) -	REGION_UNINIT(pScreen, srcReg); -    if (maskReg) -	REGION_UNINIT(pScreen, maskReg); - -    pDstPix = exaGetDrawablePixmap(pDst->pDrawable); -    if (!exaOpReadsDestination(op)) { -	int xoff; -	int yoff; - -	exaGetDrawableDeltas (pDst->pDrawable, pDstPix, &xoff, &yoff); -	REGION_TRANSLATE(pScreen, ®ion, pDst->pDrawable->x + xoff, -			 pDst->pDrawable->y + yoff); -	dstReg = ®ion; -    } - -    if (pDst->alphaMap && pDst->alphaMap->pDrawable) -	pExaScr->prepare_access_reg(exaGetDrawablePixmap(pDst->alphaMap->pDrawable), -				    EXA_PREPARE_AUX_DEST, -				    dstReg); -    pExaScr->prepare_access_reg(pDstPix, EXA_PREPARE_DEST, dstReg); - -    REGION_UNINIT(pScreen, ®ion); -    return TRUE; -} - -void -ExaCheckComposite (CARD8      op, -                   PicturePtr pSrc, -                   PicturePtr pMask, -                   PicturePtr pDst, -                   INT16      xSrc, -                   INT16      ySrc, -                   INT16      xMask, -                   INT16      yMask, -                   INT16      xDst, -                   INT16      yDst, -                   CARD16     width, -                   CARD16     height) -{ -    ScreenPtr pScreen = pDst->pDrawable->pScreen; -#ifdef RENDER -    PictureScreenPtr	ps = GetPictureScreen(pScreen); -#endif /* RENDER */ -    EXA_PRE_FALLBACK(pScreen); - -    if (pExaScr->prepare_access_reg) { -	if (!ExaPrepareCompositeReg(pScreen, op, pSrc, pMask, pDst, xSrc, -				   ySrc, xMask, yMask, xDst, yDst, width, -				   height)) -	    goto out_no_clip; -    } else { - -	/* We need to prepare access to any separate alpha maps first, -	 * in case the driver doesn't support EXA_PREPARE_AUX*, -	 * in which case EXA_PREPARE_SRC may be used for moving them out. -	 */ - -	if (pSrc->alphaMap && pSrc->alphaMap->pDrawable) -	    exaPrepareAccess(pSrc->alphaMap->pDrawable, EXA_PREPARE_AUX_SRC); -	if (pMask && pMask->alphaMap && pMask->alphaMap->pDrawable) -	    exaPrepareAccess(pMask->alphaMap->pDrawable, EXA_PREPARE_AUX_MASK); -	if (pDst->alphaMap && pDst->alphaMap->pDrawable) -	    exaPrepareAccess(pDst->alphaMap->pDrawable, EXA_PREPARE_AUX_DEST); - -	exaPrepareAccess (pDst->pDrawable, EXA_PREPARE_DEST); - -	EXA_FALLBACK(("from picts %p/%p to pict %p\n", -		      pSrc, pMask, pDst)); - -	if (pSrc->pDrawable != NULL) -	    exaPrepareAccess (pSrc->pDrawable, EXA_PREPARE_SRC); -	if (pMask && pMask->pDrawable != NULL) -	    exaPrepareAccess (pMask->pDrawable, EXA_PREPARE_MASK); -    } - -#ifdef RENDER -    swap(pExaScr, ps, Composite); -    ps->Composite (op, -                 pSrc, -                 pMask, -                 pDst, -                 xSrc, -                 ySrc, -                 xMask, -                 yMask, -                 xDst, -                 yDst, -                 width, -                 height); -    swap(pExaScr, ps, Composite); -#endif /* RENDER */ -    if (pMask && pMask->pDrawable != NULL) -	exaFinishAccess (pMask->pDrawable, EXA_PREPARE_MASK); -    if (pSrc->pDrawable != NULL) -	exaFinishAccess (pSrc->pDrawable, EXA_PREPARE_SRC); -    exaFinishAccess (pDst->pDrawable, EXA_PREPARE_DEST); -    if (pDst->alphaMap && pDst->alphaMap->pDrawable) -	exaFinishAccess(pDst->alphaMap->pDrawable, EXA_PREPARE_AUX_DEST); -    if (pSrc->alphaMap && pSrc->alphaMap->pDrawable) -	exaFinishAccess(pSrc->alphaMap->pDrawable, EXA_PREPARE_AUX_SRC); -    if (pMask && pMask->alphaMap && pMask->alphaMap->pDrawable) -	exaFinishAccess(pMask->alphaMap->pDrawable, EXA_PREPARE_AUX_MASK); - -out_no_clip: -    EXA_POST_FALLBACK(pScreen); -} - -void -ExaCheckAddTraps (PicturePtr	pPicture, -		  INT16		x_off, -		  INT16		y_off, -		  int		ntrap, -		  xTrap		*traps) -{ -    ScreenPtr pScreen = pPicture->pDrawable->pScreen; -#ifdef RENDER -    PictureScreenPtr	ps = GetPictureScreen(pScreen); -#endif /* RENDER */ -    EXA_PRE_FALLBACK(pScreen); - -    EXA_FALLBACK(("to pict %p (%c)\n", -		  exaDrawableLocation(pPicture->pDrawable))); -    exaPrepareAccess(pPicture->pDrawable, EXA_PREPARE_DEST); -#ifdef RENDER -    swap(pExaScr, ps, AddTraps); -    ps->AddTraps (pPicture, x_off, y_off, ntrap, traps); -    swap(pExaScr, ps, AddTraps); -#endif /* RENDER */ -    exaFinishAccess(pPicture->pDrawable, EXA_PREPARE_DEST); -    EXA_POST_FALLBACK(pScreen); -} - -/** - * Gets the 0,0 pixel of a pixmap.  Used for doing solid fills of tiled pixmaps - * that happen to be 1x1.  Pixmap must be at least 8bpp. - */ -CARD32 -exaGetPixmapFirstPixel (PixmapPtr pPixmap) -{ -    switch (pPixmap->drawable.bitsPerPixel) { -    case 32: -	{ -	    CARD32 pixel; - -	    pPixmap->drawable.pScreen->GetImage(&pPixmap->drawable, 0, 0, 1, 1, -						ZPixmap, ~0, (char*)&pixel); -	    return pixel; -	} -    case 16: -	{ -	    CARD16 pixel; - -	    pPixmap->drawable.pScreen->GetImage(&pPixmap->drawable, 0, 0, 1, 1, -						ZPixmap, ~0, (char*)&pixel); -	    return pixel; -	} -    case 8: -	{ -	    CARD8 pixel; - -	    pPixmap->drawable.pScreen->GetImage(&pPixmap->drawable, 0, 0, 1, 1, -						ZPixmap, ~0, (char*)&pixel); -	    return pixel; -	} -    default: -	FatalError("%s called for invalid bpp %d\n", __func__, -		   pPixmap->drawable.bitsPerPixel); -    } -} +/*
 + *
 + * Copyright © 1999 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.
 + */
 +
 +#include "exa_priv.h"
 +
 +#ifdef RENDER
 +#include "mipict.h"
 +#endif
 +
 +/*
 + * These functions wrap the low-level fb rendering functions and
 + * synchronize framebuffer/accelerated drawing by stalling until
 + * the accelerator is idle
 + */
 +
 +/**
 + * Calls exaPrepareAccess with EXA_PREPARE_SRC for the tile, if that is the
 + * current fill style.
 + *
 + * Solid doesn't use an extra pixmap source, and Stippled/OpaqueStippled are
 + * 1bpp and never in fb, so we don't worry about them.
 + * We should worry about them for completeness sake and going forward.
 + */
 +void
 +exaPrepareAccessGC(GCPtr pGC)
 +{
 +    if (pGC->stipple)
 +        exaPrepareAccess(&pGC->stipple->drawable, EXA_PREPARE_MASK);
 +    if (pGC->fillStyle == FillTiled)
 +	exaPrepareAccess(&pGC->tile.pixmap->drawable, EXA_PREPARE_SRC);
 +}
 +
 +/**
 + * Finishes access to the tile in the GC, if used.
 + */
 +void
 +exaFinishAccessGC(GCPtr pGC)
 +{
 +    if (pGC->fillStyle == FillTiled)
 +	exaFinishAccess(&pGC->tile.pixmap->drawable, EXA_PREPARE_SRC);
 +    if (pGC->stipple)
 +        exaFinishAccess(&pGC->stipple->drawable, EXA_PREPARE_MASK);
 +}
 +
 +#if DEBUG_TRACE_FALL
 +char
 +exaDrawableLocation(DrawablePtr pDrawable)
 +{
 +    return exaDrawableIsOffscreen(pDrawable) ? 's' : 'm';
 +}
 +#endif /* DEBUG_TRACE_FALL */
 +
 +void
 +ExaCheckFillSpans (DrawablePtr pDrawable, GCPtr pGC, int nspans,
 +		   DDXPointPtr ppt, int *pwidth, int fSorted)
 +{
 +    EXA_PRE_FALLBACK_GC(pGC);
 +    EXA_FALLBACK(("to %p (%c)\n", pDrawable, exaDrawableLocation(pDrawable)));
 +    exaPrepareAccess (pDrawable, EXA_PREPARE_DEST);
 +    exaPrepareAccessGC (pGC);
 +    pGC->ops->FillSpans (pDrawable, pGC, nspans, ppt, pwidth, fSorted);
 +    exaFinishAccessGC (pGC);
 +    exaFinishAccess (pDrawable, EXA_PREPARE_DEST);
 +    EXA_POST_FALLBACK_GC(pGC);
 +}
 +
 +void
 +ExaCheckSetSpans (DrawablePtr pDrawable, GCPtr pGC, char *psrc,
 +		 DDXPointPtr ppt, int *pwidth, int nspans, int fSorted)
 +{
 +    EXA_PRE_FALLBACK_GC(pGC);
 +    EXA_FALLBACK(("to %p (%c)\n", pDrawable, exaDrawableLocation(pDrawable)));
 +    exaPrepareAccess (pDrawable, EXA_PREPARE_DEST);
 +    pGC->ops->SetSpans (pDrawable, pGC, psrc, ppt, pwidth, nspans, fSorted);
 +    exaFinishAccess (pDrawable, EXA_PREPARE_DEST);
 +    EXA_POST_FALLBACK_GC(pGC);
 +}
 +
 +void
 +ExaCheckPutImage (DrawablePtr pDrawable, GCPtr pGC, int depth,
 +		 int x, int y, int w, int h, int leftPad, int format,
 +		 char *bits)
 +{
 +    PixmapPtr pPixmap = exaGetDrawablePixmap(pDrawable);
 +    ExaPixmapPriv(pPixmap);
 +
 +    EXA_PRE_FALLBACK_GC(pGC);
 +    EXA_FALLBACK(("to %p (%c)\n", pDrawable, exaDrawableLocation(pDrawable)));
 +    if (!pExaScr->prepare_access_reg || !pExaPixmap->pDamage ||
 +	exaGCReadsDestination(pDrawable, pGC->planemask, pGC->fillStyle,
 +			      pGC->alu, pGC->clientClipType))
 +	exaPrepareAccess (pDrawable, EXA_PREPARE_DEST);
 +    else
 +	pExaScr->prepare_access_reg(pPixmap, EXA_PREPARE_DEST,
 +				    DamagePendingRegion(pExaPixmap->pDamage));
 +    pGC->ops->PutImage (pDrawable, pGC, depth, x, y, w, h, leftPad, format, bits);
 +    exaFinishAccess (pDrawable, EXA_PREPARE_DEST);
 +    EXA_POST_FALLBACK_GC(pGC);
 +}
 +
 +void
 +ExaCheckCopyNtoN (DrawablePtr pSrc, DrawablePtr pDst,  GCPtr pGC,
 +	     BoxPtr	pbox, int nbox, int dx, int dy, Bool	reverse, 
 +	     Bool upsidedown, Pixel bitplane, void *closure)
 +{
 +    RegionRec reg;
 +    int xoff, yoff;
 +    EXA_PRE_FALLBACK_GC(pGC);
 +    EXA_FALLBACK(("from %p to %p (%c,%c)\n", pSrc, pDst,
 +		  exaDrawableLocation(pSrc), exaDrawableLocation(pDst)));
 +
 +    if (pExaScr->prepare_access_reg) {
 +	PixmapPtr pPixmap = exaGetDrawablePixmap(pSrc);
 +
 +	exaGetDrawableDeltas(pSrc, pPixmap, &xoff, &yoff);
 +	REGION_INIT(pScreen, ®, pbox, nbox);
 +	REGION_TRANSLATE(pScreen, ®, xoff + dx, yoff + dy);
 +	pExaScr->prepare_access_reg(pPixmap, EXA_PREPARE_SRC, ®);
 +	REGION_UNINIT(pScreen, ®);
 +    } else
 +	exaPrepareAccess (pSrc, EXA_PREPARE_SRC);
 +
 +    if (pExaScr->prepare_access_reg &&
 +	!exaGCReadsDestination(pDst, pGC->planemask, pGC->fillStyle,
 +			       pGC->alu, pGC->clientClipType)) {
 +	PixmapPtr pPixmap = exaGetDrawablePixmap(pDst);
 +
 +	exaGetDrawableDeltas(pSrc, pPixmap, &xoff, &yoff);
 +	REGION_INIT(pScreen, ®, pbox, nbox);
 +	REGION_TRANSLATE(pScreen, ®, xoff, yoff);
 +	pExaScr->prepare_access_reg(pPixmap, EXA_PREPARE_DEST, ®);
 +	REGION_UNINIT(pScreen, ®);
 +    } else
 +	exaPrepareAccess (pDst, EXA_PREPARE_DEST);
 +
 +    /* This will eventually call fbCopyNtoN, with some calculation overhead. */
 +    while (nbox--) {
 +	pGC->ops->CopyArea (pSrc, pDst, pGC, pbox->x1 - pSrc->x + dx, pbox->y1 - pSrc->y + dy, 
 +			pbox->x2 - pbox->x1, pbox->y2 - pbox->y1, pbox->x1 - pDst->x, pbox->y1 - pDst->y);
 +	pbox++;
 +    }
 +    exaFinishAccess (pSrc, EXA_PREPARE_SRC);
 +    exaFinishAccess (pDst, EXA_PREPARE_DEST);
 +    EXA_POST_FALLBACK_GC(pGC);
 +}
 +
 +static void
 +ExaFallbackPrepareReg(DrawablePtr pDrawable,
 +		      GCPtr pGC,
 +		      int x, int y, int width, int height,
 +		      int index, Bool checkReads)
 +{
 +    ScreenPtr pScreen = pDrawable->pScreen;
 +    ExaScreenPriv(pScreen);
 +
 +    if (pExaScr->prepare_access_reg &&
 +	!(checkReads && exaGCReadsDestination(pDrawable,
 +					      pGC->planemask,
 +					      pGC->fillStyle,
 +					      pGC->alu,
 +					      pGC->clientClipType))) {
 +	BoxRec box;
 +	RegionRec reg;
 +	int xoff, yoff;
 +	PixmapPtr pPixmap = exaGetDrawablePixmap(pDrawable);
 +
 +	exaGetDrawableDeltas(pDrawable, pPixmap, &xoff, &yoff);
 +	box.x1 = pDrawable->x + x + xoff;
 +	box.y1 = pDrawable->y + y + yoff;
 +	box.x2 = box.x1 + width;
 +	box.y2 = box.y1 + height;
 +
 +	REGION_INIT(pScreen, ®, &box, 1);
 +	pExaScr->prepare_access_reg(pPixmap, index, ®);
 +	REGION_UNINIT(pScreen, ®);
 +    } else
 +	exaPrepareAccess(pDrawable, index);
 +}
 +
 +
 +RegionPtr
 +ExaCheckCopyArea (DrawablePtr pSrc, DrawablePtr pDst, GCPtr pGC,
 +		 int srcx, int srcy, int w, int h, int dstx, int dsty)
 +{
 +    RegionPtr ret;
 +
 +    EXA_PRE_FALLBACK_GC(pGC);
 +    EXA_FALLBACK(("from %p to %p (%c,%c)\n", pSrc, pDst,
 +		  exaDrawableLocation(pSrc), exaDrawableLocation(pDst)));
 +    ExaFallbackPrepareReg(pSrc, pGC, srcx, srcy, w, h,
 +			  EXA_PREPARE_SRC, FALSE);
 +    ExaFallbackPrepareReg(pDst, pGC, dstx, dsty, w, h,
 +			  EXA_PREPARE_DEST, TRUE);
 +    ret = pGC->ops->CopyArea (pSrc, pDst, pGC, srcx, srcy, w, h, dstx, dsty);
 +    exaFinishAccess (pSrc, EXA_PREPARE_SRC);
 +    exaFinishAccess (pDst, EXA_PREPARE_DEST);
 +    EXA_POST_FALLBACK_GC(pGC);
 +
 +    return ret;
 +}
 +
 +RegionPtr
 +ExaCheckCopyPlane (DrawablePtr pSrc, DrawablePtr pDst, GCPtr pGC,
 +		  int srcx, int srcy, int w, int h, int dstx, int dsty,
 +		  unsigned long bitPlane)
 +{
 +    RegionPtr ret;
 +
 +    EXA_PRE_FALLBACK_GC(pGC);
 +    EXA_FALLBACK(("from %p to %p (%c,%c)\n", pSrc, pDst,
 +		  exaDrawableLocation(pSrc), exaDrawableLocation(pDst)));
 +    ExaFallbackPrepareReg(pSrc, pGC, srcx, srcy, w, h,
 +			  EXA_PREPARE_SRC, FALSE);
 +    ExaFallbackPrepareReg(pDst, pGC, dstx, dsty, w, h,
 +			  EXA_PREPARE_DEST, TRUE);
 +    ret = pGC->ops->CopyPlane (pSrc, pDst, pGC, srcx, srcy, w, h, dstx, dsty,
 +		       bitPlane);
 +    exaFinishAccess (pSrc, EXA_PREPARE_SRC);
 +    exaFinishAccess (pDst, EXA_PREPARE_DEST);
 +    EXA_POST_FALLBACK_GC(pGC);
 +
 +    return ret;
 +}
 +
 +void
 +ExaCheckPolyPoint (DrawablePtr pDrawable, GCPtr pGC, int mode, int npt,
 +		  DDXPointPtr pptInit)
 +{
 +    EXA_PRE_FALLBACK_GC(pGC);
 +    EXA_FALLBACK(("to %p (%c)\n", pDrawable, exaDrawableLocation(pDrawable)));
 +    exaPrepareAccess (pDrawable, EXA_PREPARE_DEST);
 +    pGC->ops->PolyPoint (pDrawable, pGC, mode, npt, pptInit);
 +    exaFinishAccess (pDrawable, EXA_PREPARE_DEST);
 +    EXA_POST_FALLBACK_GC(pGC);
 +}
 +
 +void
 +ExaCheckPolylines (DrawablePtr pDrawable, GCPtr pGC,
 +		  int mode, int npt, DDXPointPtr ppt)
 +{
 +    EXA_PRE_FALLBACK_GC(pGC);
 +    EXA_FALLBACK(("to %p (%c), width %d, mode %d, count %d\n",
 +		  pDrawable, exaDrawableLocation(pDrawable),
 +		  pGC->lineWidth, mode, npt));
 +
 +    exaPrepareAccess (pDrawable, EXA_PREPARE_DEST);
 +    exaPrepareAccessGC (pGC);
 +    pGC->ops->Polylines (pDrawable, pGC, mode, npt, ppt);
 +    exaFinishAccessGC (pGC);
 +    exaFinishAccess (pDrawable, EXA_PREPARE_DEST);
 +    EXA_POST_FALLBACK_GC(pGC);
 +}
 +
 +void
 +ExaCheckPolySegment (DrawablePtr pDrawable, GCPtr pGC,
 +		    int nsegInit, xSegment *pSegInit)
 +{
 +    EXA_PRE_FALLBACK_GC(pGC);
 +    EXA_FALLBACK(("to %p (%c) width %d, count %d\n", pDrawable,
 +		  exaDrawableLocation(pDrawable), pGC->lineWidth, nsegInit));
 +
 +    exaPrepareAccess (pDrawable, EXA_PREPARE_DEST);
 +    exaPrepareAccessGC (pGC);
 +    pGC->ops->PolySegment (pDrawable, pGC, nsegInit, pSegInit);
 +    exaFinishAccessGC (pGC);
 +    exaFinishAccess (pDrawable, EXA_PREPARE_DEST);
 +    EXA_POST_FALLBACK_GC(pGC);
 +}
 +
 +void
 +ExaCheckPolyArc (DrawablePtr pDrawable, GCPtr pGC,
 +		int narcs, xArc *pArcs)
 +{
 +    EXA_PRE_FALLBACK_GC(pGC);
 +    EXA_FALLBACK(("to %p (%c)\n", pDrawable, exaDrawableLocation(pDrawable)));
 +
 +    exaPrepareAccess (pDrawable, EXA_PREPARE_DEST);
 +    exaPrepareAccessGC (pGC);
 +    pGC->ops->PolyArc (pDrawable, pGC, narcs, pArcs);
 +    exaFinishAccessGC (pGC);
 +    exaFinishAccess (pDrawable, EXA_PREPARE_DEST);
 +    EXA_POST_FALLBACK_GC(pGC);
 +}
 +
 +void
 +ExaCheckPolyFillRect (DrawablePtr pDrawable, GCPtr pGC,
 +		     int nrect, xRectangle *prect)
 +{
 +    EXA_PRE_FALLBACK_GC(pGC);
 +    EXA_FALLBACK(("to %p (%c)\n", pDrawable, exaDrawableLocation(pDrawable)));
 +
 +    exaPrepareAccess (pDrawable, EXA_PREPARE_DEST);
 +    exaPrepareAccessGC (pGC);
 +    pGC->ops->PolyFillRect (pDrawable, pGC, nrect, prect);
 +    exaFinishAccessGC (pGC);
 +    exaFinishAccess (pDrawable, EXA_PREPARE_DEST);
 +    EXA_POST_FALLBACK_GC(pGC);
 +}
 +
 +void
 +ExaCheckImageGlyphBlt (DrawablePtr pDrawable, GCPtr pGC,
 +		      int x, int y, unsigned int nglyph,
 +		      CharInfoPtr *ppci, pointer pglyphBase)
 +{
 +    EXA_PRE_FALLBACK_GC(pGC);
 +    EXA_FALLBACK(("to %p (%c)\n", pDrawable,
 +		  exaDrawableLocation(pDrawable)));
 +    exaPrepareAccess (pDrawable, EXA_PREPARE_DEST);
 +    exaPrepareAccessGC (pGC);
 +    pGC->ops->ImageGlyphBlt (pDrawable, pGC, x, y, nglyph, ppci, pglyphBase);
 +    exaFinishAccessGC (pGC);
 +    exaFinishAccess (pDrawable, EXA_PREPARE_DEST);
 +    EXA_POST_FALLBACK_GC(pGC);
 +}
 +
 +void
 +ExaCheckPolyGlyphBlt (DrawablePtr pDrawable, GCPtr pGC,
 +		     int x, int y, unsigned int nglyph,
 +		     CharInfoPtr *ppci, pointer pglyphBase)
 +{
 +    EXA_PRE_FALLBACK_GC(pGC);
 +    EXA_FALLBACK(("to %p (%c), style %d alu %d\n", pDrawable,
 +		  exaDrawableLocation(pDrawable), pGC->fillStyle, pGC->alu));
 +    exaPrepareAccess (pDrawable, EXA_PREPARE_DEST);
 +    exaPrepareAccessGC (pGC);
 +    pGC->ops->PolyGlyphBlt (pDrawable, pGC, x, y, nglyph, ppci, pglyphBase);
 +    exaFinishAccessGC (pGC);
 +    exaFinishAccess (pDrawable, EXA_PREPARE_DEST);
 +    EXA_POST_FALLBACK_GC(pGC);
 +}
 +
 +void
 +ExaCheckPushPixels (GCPtr pGC, PixmapPtr pBitmap,
 +		   DrawablePtr pDrawable,
 +		   int w, int h, int x, int y)
 +{
 +    EXA_PRE_FALLBACK_GC(pGC);
 +    EXA_FALLBACK(("from %p to %p (%c,%c)\n", pBitmap, pDrawable,
 +		  exaDrawableLocation(&pBitmap->drawable),
 +		  exaDrawableLocation(pDrawable)));
 +    ExaFallbackPrepareReg(pDrawable, pGC, x, y, w, h,
 +			  EXA_PREPARE_DEST, TRUE);
 +    ExaFallbackPrepareReg(&pBitmap->drawable, pGC, 0, 0, w, h,
 +			  EXA_PREPARE_SRC, FALSE);
 +    exaPrepareAccessGC (pGC);
 +    pGC->ops->PushPixels (pGC, pBitmap, pDrawable, w, h, x, y);
 +    exaFinishAccessGC (pGC);
 +    exaFinishAccess (&pBitmap->drawable, EXA_PREPARE_SRC);
 +    exaFinishAccess (pDrawable, EXA_PREPARE_DEST);
 +    EXA_POST_FALLBACK_GC(pGC);
 +}
 +
 +void
 +ExaCheckCopyWindow(WindowPtr pWin, DDXPointRec ptOldOrg, RegionPtr prgnSrc)
 +{
 +    DrawablePtr pDrawable = &pWin->drawable;
 +    ScreenPtr pScreen = pDrawable->pScreen;
 +    EXA_PRE_FALLBACK(pScreen);
 +    EXA_FALLBACK(("from %p\n", pWin));
 +
 +    /* Only need the source bits, the destination region will be overwritten */
 +    if (pExaScr->prepare_access_reg) {
 +	PixmapPtr pPixmap = pScreen->GetWindowPixmap(pWin);
 +	int xoff, yoff;
 +
 +	exaGetDrawableDeltas(&pWin->drawable, pPixmap, &xoff, &yoff);
 +	REGION_TRANSLATE(pScreen, prgnSrc, xoff, yoff);
 +	pExaScr->prepare_access_reg(pPixmap, EXA_PREPARE_SRC, prgnSrc);
 +	REGION_TRANSLATE(pScreen, prgnSrc, -xoff, -yoff);
 +    } else
 +	exaPrepareAccess(pDrawable, EXA_PREPARE_SRC);
 +
 +    swap(pExaScr, pScreen, CopyWindow);
 +    pScreen->CopyWindow (pWin, ptOldOrg, prgnSrc);
 +    swap(pExaScr, pScreen, CopyWindow);
 +    exaFinishAccess (pDrawable, EXA_PREPARE_SRC);
 +    EXA_POST_FALLBACK(pScreen);
 +}
 +
 +void
 +ExaCheckGetImage(DrawablePtr pDrawable, int x, int y, int w, int h,
 +		unsigned int format, unsigned long planeMask, char *d)
 +{
 +    ScreenPtr pScreen = pDrawable->pScreen;
 +    EXA_PRE_FALLBACK(pScreen);
 +    EXA_FALLBACK(("from %p (%c)\n", pDrawable,
 +		  exaDrawableLocation(pDrawable)));
 +
 +    ExaFallbackPrepareReg(pDrawable, NULL, x, y, w, h,
 +			  EXA_PREPARE_SRC, FALSE);
 +    swap(pExaScr, pScreen, GetImage);
 +    pScreen->GetImage (pDrawable, x, y, w, h, format, planeMask, d);
 +    swap(pExaScr, pScreen, GetImage);
 +    exaFinishAccess (pDrawable, EXA_PREPARE_SRC);
 +    EXA_POST_FALLBACK(pScreen);
 +}
 +
 +void
 +ExaCheckGetSpans (DrawablePtr pDrawable,
 +		 int wMax,
 +		 DDXPointPtr ppt,
 +		 int *pwidth,
 +		 int nspans,
 +		 char *pdstStart)
 +{
 +    ScreenPtr pScreen = pDrawable->pScreen;
 +
 +    EXA_PRE_FALLBACK(pScreen);
 +    EXA_FALLBACK(("from %p (%c)\n", pDrawable, exaDrawableLocation(pDrawable)));
 +    exaPrepareAccess (pDrawable, EXA_PREPARE_SRC);
 +    swap(pExaScr, pScreen, GetSpans);
 +    pScreen->GetSpans (pDrawable, wMax, ppt, pwidth, nspans, pdstStart);
 +    swap(pExaScr, pScreen, GetSpans);
 +    exaFinishAccess (pDrawable, EXA_PREPARE_SRC);
 +    EXA_POST_FALLBACK(pScreen);
 +}
 +
 +static void
 +ExaSrcValidate(DrawablePtr pDrawable,
 +	       int x,
 +	       int y,
 +	       int width,
 +	       int height)
 +{
 +    ScreenPtr pScreen = pDrawable->pScreen;
 +    ExaScreenPriv(pScreen);
 +    PixmapPtr pPix = exaGetDrawablePixmap (pDrawable);
 +    BoxRec box;
 +    RegionRec reg;
 +    RegionPtr dst;
 +    int xoff, yoff;
 +
 +    exaGetDrawableDeltas(pDrawable, pPix, &xoff, &yoff);
 +
 +    box.x1 = x + xoff;
 +    box.y1 = y + yoff;
 +    box.x2 = box.x1 + width;
 +    box.y2 = box.y1 + height;
 +
 +    dst = (pExaScr->srcPix == pPix) ? &pExaScr->srcReg :
 +	&pExaScr->maskReg;
 +
 +    REGION_INIT(pScreen, ®, &box, 1);
 +    REGION_UNION(pScreen, dst, dst, ®);
 +    REGION_UNINIT(pScreen, ®);
 +
 +    swap(pExaScr, pScreen, SourceValidate);
 +    pScreen->SourceValidate(pDrawable, x, y, width, height);
 +    swap(pExaScr, pScreen, SourceValidate);
 +}
 +
 +static Bool
 +ExaPrepareCompositeReg(ScreenPtr  pScreen,
 +		       CARD8      op,
 +		       PicturePtr pSrc,
 +		       PicturePtr pMask,
 +		       PicturePtr pDst,
 +		       INT16      xSrc,
 +		       INT16      ySrc,
 +		       INT16      xMask,
 +		       INT16      yMask,
 +		       INT16      xDst,
 +		       INT16      yDst,
 +		       CARD16     width,
 +		       CARD16     height)
 +{
 +    RegionRec region;
 +    RegionPtr dstReg = NULL;
 +    RegionPtr srcReg = NULL;
 +    RegionPtr maskReg = NULL;
 +    PixmapPtr pSrcPix = NULL;
 +    PixmapPtr pMaskPix = NULL;
 +    PixmapPtr pDstPix;
 +    ExaScreenPriv(pScreen);
 +    Bool ret;
 +
 +
 +    REGION_NULL(pScreen, ®ion);
 +
 +    if (pSrc->pDrawable) {
 +	pSrcPix = exaGetDrawablePixmap(pSrc->pDrawable);
 +	REGION_NULL(pScreen, &pExaScr->srcReg);
 +	srcReg = &pExaScr->srcReg;
 +	pExaScr->srcPix = pSrcPix;
 +	if (pSrc != pDst)
 +	    REGION_TRANSLATE(pScreen, pSrc->pCompositeClip,
 +			     -pSrc->pDrawable->x,
 +			     -pSrc->pDrawable->y);
 +    }
 +
 +    if (pMask && pMask->pDrawable) {
 +	pMaskPix = exaGetDrawablePixmap(pMask->pDrawable);
 +	REGION_NULL(pScreen, &pExaScr->maskReg);
 +	maskReg = &pExaScr->maskReg;
 +	if (pMask != pDst && pMask != pSrc)
 +	    REGION_TRANSLATE(pScreen, pMask->pCompositeClip,
 +			     -pMask->pDrawable->x,
 +			     -pMask->pDrawable->y);
 +    }
 +
 +    REGION_TRANSLATE(pScreen, pDst->pCompositeClip,
 +		     -pDst->pDrawable->x,
 +		     -pDst->pDrawable->y);
 +
 +    pExaScr->SavedSourceValidate = ExaSrcValidate;
 +    swap(pExaScr, pScreen, SourceValidate);
 +    ret = miComputeCompositeRegion (®ion, pSrc, pMask, pDst,
 +				    xSrc, ySrc, xMask, yMask,
 +				    xDst,
 +				    yDst,
 +				    width, height);
 +    swap(pExaScr, pScreen, SourceValidate);
 +
 +    REGION_TRANSLATE(pScreen, pDst->pCompositeClip,
 +		     pDst->pDrawable->x,
 +		     pDst->pDrawable->y);
 +    if (pSrc->pDrawable && pSrc != pDst)
 +	REGION_TRANSLATE(pScreen, pSrc->pCompositeClip,
 +			 pSrc->pDrawable->x,
 +			 pSrc->pDrawable->y);
 +    if (pMask && pMask->pDrawable && pMask != pDst && pMask != pSrc)
 +	REGION_TRANSLATE(pScreen, pMask->pCompositeClip,
 +			 pMask->pDrawable->x,
 +			 pMask->pDrawable->y);
 +
 +    if (!ret) {
 +	if (srcReg)
 +	    REGION_UNINIT(pScreen, srcReg);
 +	if (maskReg)
 +	    REGION_UNINIT(pScreen, maskReg);
 +
 +	return FALSE;
 +    }
 +
 +    /**
 +     * Don't limit alphamaps readbacks for now until we've figured out how that
 +     * should be done.
 +     */
 +
 +    if (pSrc->alphaMap && pSrc->alphaMap->pDrawable)
 +	pExaScr->prepare_access_reg(exaGetDrawablePixmap(pSrc->alphaMap->pDrawable),
 +				    EXA_PREPARE_AUX_SRC,
 +				    NULL);
 +    if (pMask && pMask->alphaMap && pMask->alphaMap->pDrawable)
 +	pExaScr->prepare_access_reg(exaGetDrawablePixmap(pMask->alphaMap->pDrawable),
 +				    EXA_PREPARE_AUX_MASK,
 +				    NULL);
 +
 +    if (pSrcPix)
 +	pExaScr->prepare_access_reg(pSrcPix,
 +				    EXA_PREPARE_SRC,
 +				    srcReg);
 +
 +    if (pMaskPix)
 +	pExaScr->prepare_access_reg(pMaskPix,
 +				    EXA_PREPARE_MASK,
 +				    maskReg);
 +
 +    if (srcReg)
 +	REGION_UNINIT(pScreen, srcReg);
 +    if (maskReg)
 +	REGION_UNINIT(pScreen, maskReg);
 +
 +    pDstPix = exaGetDrawablePixmap(pDst->pDrawable);
 +    if (!exaOpReadsDestination(op)) {
 +	int xoff;
 +	int yoff;
 +
 +	exaGetDrawableDeltas (pDst->pDrawable, pDstPix, &xoff, &yoff);
 +	REGION_TRANSLATE(pScreen, ®ion, pDst->pDrawable->x + xoff,
 +			 pDst->pDrawable->y + yoff);
 +	dstReg = ®ion;
 +    }
 +
 +    if (pDst->alphaMap && pDst->alphaMap->pDrawable)
 +	pExaScr->prepare_access_reg(exaGetDrawablePixmap(pDst->alphaMap->pDrawable),
 +				    EXA_PREPARE_AUX_DEST,
 +				    dstReg);
 +    pExaScr->prepare_access_reg(pDstPix, EXA_PREPARE_DEST, dstReg);
 +
 +    REGION_UNINIT(pScreen, ®ion);
 +    return TRUE;
 +}
 +
 +void
 +ExaCheckComposite (CARD8      op,
 +                   PicturePtr pSrc,
 +                   PicturePtr pMask,
 +                   PicturePtr pDst,
 +                   INT16      xSrc,
 +                   INT16      ySrc,
 +                   INT16      xMask,
 +                   INT16      yMask,
 +                   INT16      xDst,
 +                   INT16      yDst,
 +                   CARD16     width,
 +                   CARD16     height)
 +{
 +    ScreenPtr pScreen = pDst->pDrawable->pScreen;
 +#ifdef RENDER
 +    PictureScreenPtr	ps = GetPictureScreen(pScreen);
 +#endif /* RENDER */
 +    EXA_PRE_FALLBACK(pScreen);
 +
 +    if (pExaScr->prepare_access_reg) {
 +	if (!ExaPrepareCompositeReg(pScreen, op, pSrc, pMask, pDst, xSrc,
 +				   ySrc, xMask, yMask, xDst, yDst, width,
 +				   height))
 +	    goto out_no_clip;
 +    } else {
 +
 +	/* We need to prepare access to any separate alpha maps first,
 +	 * in case the driver doesn't support EXA_PREPARE_AUX*,
 +	 * in which case EXA_PREPARE_SRC may be used for moving them out.
 +	 */
 +
 +	if (pSrc->alphaMap && pSrc->alphaMap->pDrawable)
 +	    exaPrepareAccess(pSrc->alphaMap->pDrawable, EXA_PREPARE_AUX_SRC);
 +	if (pMask && pMask->alphaMap && pMask->alphaMap->pDrawable)
 +	    exaPrepareAccess(pMask->alphaMap->pDrawable, EXA_PREPARE_AUX_MASK);
 +	if (pDst->alphaMap && pDst->alphaMap->pDrawable)
 +	    exaPrepareAccess(pDst->alphaMap->pDrawable, EXA_PREPARE_AUX_DEST);
 +
 +	exaPrepareAccess (pDst->pDrawable, EXA_PREPARE_DEST);
 +
 +	EXA_FALLBACK(("from picts %p/%p to pict %p\n",
 +		      pSrc, pMask, pDst));
 +
 +	if (pSrc->pDrawable != NULL)
 +	    exaPrepareAccess (pSrc->pDrawable, EXA_PREPARE_SRC);
 +	if (pMask && pMask->pDrawable != NULL)
 +	    exaPrepareAccess (pMask->pDrawable, EXA_PREPARE_MASK);
 +    }
 +
 +#ifdef RENDER
 +    swap(pExaScr, ps, Composite);
 +    ps->Composite (op,
 +                 pSrc,
 +                 pMask,
 +                 pDst,
 +                 xSrc,
 +                 ySrc,
 +                 xMask,
 +                 yMask,
 +                 xDst,
 +                 yDst,
 +                 width,
 +                 height);
 +    swap(pExaScr, ps, Composite);
 +#endif /* RENDER */
 +    if (pMask && pMask->pDrawable != NULL)
 +	exaFinishAccess (pMask->pDrawable, EXA_PREPARE_MASK);
 +    if (pSrc->pDrawable != NULL)
 +	exaFinishAccess (pSrc->pDrawable, EXA_PREPARE_SRC);
 +    exaFinishAccess (pDst->pDrawable, EXA_PREPARE_DEST);
 +    if (pDst->alphaMap && pDst->alphaMap->pDrawable)
 +	exaFinishAccess(pDst->alphaMap->pDrawable, EXA_PREPARE_AUX_DEST);
 +    if (pSrc->alphaMap && pSrc->alphaMap->pDrawable)
 +	exaFinishAccess(pSrc->alphaMap->pDrawable, EXA_PREPARE_AUX_SRC);
 +    if (pMask && pMask->alphaMap && pMask->alphaMap->pDrawable)
 +	exaFinishAccess(pMask->alphaMap->pDrawable, EXA_PREPARE_AUX_MASK);
 +
 +out_no_clip:
 +    EXA_POST_FALLBACK(pScreen);
 +}
 +
 +void
 +ExaCheckAddTraps (PicturePtr	pPicture,
 +		  INT16		x_off,
 +		  INT16		y_off,
 +		  int		ntrap,
 +		  xTrap		*traps)
 +{
 +    ScreenPtr pScreen = pPicture->pDrawable->pScreen;
 +#ifdef RENDER
 +    PictureScreenPtr	ps = GetPictureScreen(pScreen);
 +#endif /* RENDER */
 +    EXA_PRE_FALLBACK(pScreen);
 +
 +    EXA_FALLBACK(("to pict %p (%c)\n",
 +		  exaDrawableLocation(pPicture->pDrawable)));
 +    exaPrepareAccess(pPicture->pDrawable, EXA_PREPARE_DEST);
 +#ifdef RENDER
 +    swap(pExaScr, ps, AddTraps);
 +    ps->AddTraps (pPicture, x_off, y_off, ntrap, traps);
 +    swap(pExaScr, ps, AddTraps);
 +#endif /* RENDER */
 +    exaFinishAccess(pPicture->pDrawable, EXA_PREPARE_DEST);
 +    EXA_POST_FALLBACK(pScreen);
 +}
 +
 +/**
 + * Gets the 0,0 pixel of a pixmap.  Used for doing solid fills of tiled pixmaps
 + * that happen to be 1x1.  Pixmap must be at least 8bpp.
 + */
 +CARD32
 +exaGetPixmapFirstPixel (PixmapPtr pPixmap)
 +{
 +    switch (pPixmap->drawable.bitsPerPixel) {
 +    case 32:
 +	{
 +	    CARD32 pixel;
 +
 +	    pPixmap->drawable.pScreen->GetImage(&pPixmap->drawable, 0, 0, 1, 1,
 +						ZPixmap, ~0, (char*)&pixel);
 +	    return pixel;
 +	}
 +    case 16:
 +	{
 +	    CARD16 pixel;
 +
 +	    pPixmap->drawable.pScreen->GetImage(&pPixmap->drawable, 0, 0, 1, 1,
 +						ZPixmap, ~0, (char*)&pixel);
 +	    return pixel;
 +	}
 +    case 8:
 +	{
 +	    CARD8 pixel;
 +
 +	    pPixmap->drawable.pScreen->GetImage(&pPixmap->drawable, 0, 0, 1, 1,
 +						ZPixmap, ~0, (char*)&pixel);
 +	    return pixel;
 +	}
 +    default:
 +	FatalError("%s called for invalid bpp %d\n", __FUNCTION__,
 +		   pPixmap->drawable.bitsPerPixel);
 +    }
 +}
 diff --git a/xorg-server/exa/makefile b/xorg-server/exa/makefile new file mode 100644 index 000000000..5ad11a1ed --- /dev/null +++ b/xorg-server/exa/makefile @@ -0,0 +1,14 @@ +LIBRARY=libexa
 +
 +CSRCS = \
 +	exa.c \
 +	exa_classic.c \
 +	exa_migration_classic.c \
 +	exa_driver.c \
 +	exa_mixed.c \
 +	exa_migration_mixed.c \
 +	exa_accel.c \
 +	exa_glyphs.c \
 +	exa_offscreen.c \
 +	exa_render.c \
 +	exa_unaccel.c
 | 
