8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync/*
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync *
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync * Copyright © 2000 Keith Packard
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync *
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync * Permission to use, copy, modify, distribute, and sell this software and its
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync * documentation for any purpose is hereby granted without fee, provided that
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync * the above copyright notice appear in all copies and that both that
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync * copyright notice and this permission notice appear in supporting
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync * documentation, and that the name of Keith Packard not be used in
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync * advertising or publicity pertaining to distribution of the software without
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync * specific, written prior permission. Keith Packard makes no
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync * representations about the suitability of this software for any purpose. It
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync * is provided "as is" without express or implied warranty.
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync *
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync * KEITH PACKARD DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync * EVENT SHALL KEITH PACKARD BE LIABLE FOR ANY SPECIAL, INDIRECT OR
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync * DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync * PERFORMANCE OF THIS SOFTWARE.
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync */
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync/*
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync * Thanks to Daniel Chemko <dchemko@intrinsyc.com> for making the 90 and 180
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync * orientations work.
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync */
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#ifdef HAVE_DIX_CONFIG_H
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#include <dix-config.h>
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#endif
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#include <stdlib.h>
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#include <X11/X.h>
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#include "scrnintstr.h"
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#include "windowstr.h"
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#include <X11/fonts/font.h>
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#include "dixfontstr.h"
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#include <X11/fonts/fontstruct.h>
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#include "mi.h"
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#include "regionstr.h"
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#include "globals.h"
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#include "gcstruct.h"
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#include "shadow.h"
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#include "fb.h"
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define DANDEBUG 0
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#if ROTATE == 270
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define SCRLEFT(x,y,w,h) (pScreen->height - ((y) + (h)))
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define SCRY(x,y,w,h) (x)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define SCRWIDTH(x,y,w,h) (h)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define FIRSTSHA(x,y,w,h) (((y) + (h) - 1) * shaStride + (x))
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define STEPDOWN(x,y,w,h) ((w)--)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define NEXTY(x,y,w,h) ((x)++)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define SHASTEPX(stride) -(stride)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define SHASTEPY(stride) (1)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#elif ROTATE == 90
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define SCRLEFT(x,y,w,h) (y)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define SCRY(x,y,w,h) (pScreen->width - ((x) + (w)) - 1)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define SCRWIDTH(x,y,w,h) (h)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define FIRSTSHA(x,y,w,h) ((y) * shaStride + (x + w - 1))
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define STEPDOWN(x,y,w,h) ((w)--)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define NEXTY(x,y,w,h) ((void)(x))
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define SHASTEPX(stride) (stride)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define SHASTEPY(stride) (-1)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#elif ROTATE == 180
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define SCRLEFT(x,y,w,h) (pScreen->width - ((x) + (w)))
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define SCRY(x,y,w,h) (pScreen->height - ((y) + (h)) - 1)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define SCRWIDTH(x,y,w,h) (w)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define FIRSTSHA(x,y,w,h) ((y + h - 1) * shaStride + (x + w - 1))
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define STEPDOWN(x,y,w,h) ((h)--)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define NEXTY(x,y,w,h) ((void)(y))
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define SHASTEPX(stride) (-1)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define SHASTEPY(stride) -(stride)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#else
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define SCRLEFT(x,y,w,h) (x)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define SCRY(x,y,w,h) (y)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define SCRWIDTH(x,y,w,h) (w)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define FIRSTSHA(x,y,w,h) ((y) * shaStride + (x))
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define STEPDOWN(x,y,w,h) ((h)--)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define NEXTY(x,y,w,h) ((y)++)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define SHASTEPX(stride) (1)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#define SHASTEPY(stride) (stride)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#endif
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsyncvoid
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsyncFUNC (ScreenPtr pScreen,
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync shadowBufPtr pBuf)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync{
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync RegionPtr damage = shadowDamage (pBuf);
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync PixmapPtr pShadow = pBuf->pPixmap;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync int nbox = RegionNumRects (damage);
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync BoxPtr pbox = RegionRects (damage);
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync FbBits *shaBits;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync Data *shaBase, *shaLine, *sha;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync FbStride shaStride;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync int scrBase, scrLine, scr;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync int shaBpp;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync int shaXoff, shaYoff; /* XXX assumed to be zero */
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync int x, y, w, h, width;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync int i;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync Data *winBase = NULL, *win;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync CARD32 winSize;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync fbGetDrawable (&pShadow->drawable, shaBits, shaStride, shaBpp, shaXoff, shaYoff);
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync shaBase = (Data *) shaBits;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync shaStride = shaStride * sizeof (FbBits) / sizeof (Data);
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#if (DANDEBUG > 1)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync ErrorF ("-> Entering Shadow Update:\r\n |- Origins: pShadow=%x, pScreen=%x, damage=%x\r\n |- Metrics: shaStride=%d, shaBase=%x, shaBpp=%d\r\n | \n", pShadow, pScreen, damage, shaStride, shaBase, shaBpp);
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#endif
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync while (nbox--)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync {
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync x = pbox->x1;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync y = pbox->y1;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync w = (pbox->x2 - pbox->x1);
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync h = pbox->y2 - pbox->y1;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#if (DANDEBUG > 2)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync ErrorF (" |-> Redrawing box - Metrics: X=%d, Y=%d, Width=%d, Height=%d\n", x, y, w, h);
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#endif
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync scrLine = SCRLEFT(x,y,w,h);
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync shaLine = shaBase + FIRSTSHA(x,y,w,h);
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync while (STEPDOWN(x,y,w,h))
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync {
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync winSize = 0;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync scrBase = 0;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync width = SCRWIDTH(x,y,w,h);
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync scr = scrLine;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync sha = shaLine;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#if (DANDEBUG > 3)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync ErrorF (" | |-> StepDown - Metrics: width=%d, scr=%x, sha=%x\n", width, scr, sha);
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#endif
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync while (width)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync {
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync /* how much remains in this window */
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync i = scrBase + winSize - scr;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync if (i <= 0 || scr < scrBase)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync {
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync winBase = (Data *) (*pBuf->window) (pScreen,
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync SCRY(x,y,w,h),
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync scr * sizeof (Data),
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync SHADOW_WINDOW_WRITE,
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync &winSize,
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync pBuf->closure);
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync if(!winBase)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync return;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync scrBase = scr;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync winSize /= sizeof (Data);
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync i = winSize;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#if(DANDEBUG > 4)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync ErrorF (" | | |-> Starting New Line - Metrics: winBase=%x, scrBase=%x, winSize=%d\r\n | | | Xstride=%d, Ystride=%d, w=%d h=%d\n", winBase, scrBase, winSize, SHASTEPX(shaStride), SHASTEPY(shaStride), w, h);
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#endif
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync }
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync win = winBase + (scr - scrBase);
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync if (i > width)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync i = width;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync width -= i;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync scr += i;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#if(DANDEBUG > 5)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync ErrorF (" | | |-> Writing Line - Metrics: win=%x, sha=%x\n", win, sha);
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#endif
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync while (i--)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync {
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#if(DANDEBUG > 6)
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync ErrorF (" | | |-> Writing Pixel - Metrics: win=%x, sha=%d, remaining=%d\n", win, sha, i);
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync#endif
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync *win++ = *sha;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync sha += SHASTEPX(shaStride);
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync } /* i */
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync } /* width */
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync shaLine += SHASTEPY(shaStride);
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync NEXTY(x,y,w,h);
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync } /* STEPDOWN */
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync pbox++;
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync } /* nbox */
8c3d541c7ee7442f570b22eca3d82808b35b0385vboxsync}