1706f2543Smrg/* 2706f2543Smrg * 3706f2543Smrg * Copyright © 2000 Keith Packard 4706f2543Smrg * 5706f2543Smrg * Permission to use, copy, modify, distribute, and sell this software and its 6706f2543Smrg * documentation for any purpose is hereby granted without fee, provided that 7706f2543Smrg * the above copyright notice appear in all copies and that both that 8706f2543Smrg * copyright notice and this permission notice appear in supporting 9706f2543Smrg * documentation, and that the name of Keith Packard not be used in 10706f2543Smrg * advertising or publicity pertaining to distribution of the software without 11706f2543Smrg * specific, written prior permission. Keith Packard makes no 12706f2543Smrg * representations about the suitability of this software for any purpose. It 13706f2543Smrg * is provided "as is" without express or implied warranty. 14706f2543Smrg * 15706f2543Smrg * KEITH PACKARD DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, 16706f2543Smrg * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO 17706f2543Smrg * EVENT SHALL KEITH PACKARD BE LIABLE FOR ANY SPECIAL, INDIRECT OR 18706f2543Smrg * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, 19706f2543Smrg * DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER 20706f2543Smrg * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR 21706f2543Smrg * PERFORMANCE OF THIS SOFTWARE. 22706f2543Smrg */ 23706f2543Smrg 24706f2543Smrg/* 25706f2543Smrg * Thanks to Daniel Chemko <dchemko@intrinsyc.com> for making the 90 and 180 26706f2543Smrg * orientations work. 27706f2543Smrg */ 28706f2543Smrg 29706f2543Smrg#ifdef HAVE_DIX_CONFIG_H 30706f2543Smrg#include <dix-config.h> 31706f2543Smrg#endif 32706f2543Smrg 33706f2543Smrg#include <stdlib.h> 34706f2543Smrg 35706f2543Smrg#include <X11/X.h> 36706f2543Smrg#include "scrnintstr.h" 37706f2543Smrg#include "windowstr.h" 38706f2543Smrg#include <X11/fonts/font.h> 39706f2543Smrg#include "dixfontstr.h" 40706f2543Smrg#include <X11/fonts/fontstruct.h> 41706f2543Smrg#include "mi.h" 42706f2543Smrg#include "regionstr.h" 43706f2543Smrg#include "globals.h" 44706f2543Smrg#include "gcstruct.h" 45706f2543Smrg#include "shadow.h" 46706f2543Smrg#include "fb.h" 47706f2543Smrg 48706f2543Smrg#define DANDEBUG 0 49706f2543Smrg 50706f2543Smrg#if ROTATE == 270 51706f2543Smrg 52706f2543Smrg#define SCRLEFT(x,y,w,h) (pScreen->height - ((y) + (h))) 53706f2543Smrg#define SCRY(x,y,w,h) (x) 54706f2543Smrg#define SCRWIDTH(x,y,w,h) (h) 55706f2543Smrg#define FIRSTSHA(x,y,w,h) (((y) + (h) - 1) * shaStride + (x)) 56706f2543Smrg#define STEPDOWN(x,y,w,h) ((w)--) 57706f2543Smrg#define NEXTY(x,y,w,h) ((x)++) 58706f2543Smrg#define SHASTEPX(stride) -(stride) 59706f2543Smrg#define SHASTEPY(stride) (1) 60706f2543Smrg 61706f2543Smrg#elif ROTATE == 90 62706f2543Smrg 63706f2543Smrg#define SCRLEFT(x,y,w,h) (y) 64706f2543Smrg#define SCRY(x,y,w,h) (pScreen->width - ((x) + (w)) - 1) 65706f2543Smrg#define SCRWIDTH(x,y,w,h) (h) 66706f2543Smrg#define FIRSTSHA(x,y,w,h) ((y) * shaStride + (x + w - 1)) 67706f2543Smrg#define STEPDOWN(x,y,w,h) ((w)--) 68706f2543Smrg#define NEXTY(x,y,w,h) ((void)(x)) 69706f2543Smrg#define SHASTEPX(stride) (stride) 70706f2543Smrg#define SHASTEPY(stride) (-1) 71706f2543Smrg 72706f2543Smrg#elif ROTATE == 180 73706f2543Smrg 74706f2543Smrg#define SCRLEFT(x,y,w,h) (pScreen->width - ((x) + (w))) 75706f2543Smrg#define SCRY(x,y,w,h) (pScreen->height - ((y) + (h)) - 1) 76706f2543Smrg#define SCRWIDTH(x,y,w,h) (w) 77706f2543Smrg#define FIRSTSHA(x,y,w,h) ((y + h - 1) * shaStride + (x + w - 1)) 78706f2543Smrg#define STEPDOWN(x,y,w,h) ((h)--) 79706f2543Smrg#define NEXTY(x,y,w,h) ((void)(y)) 80706f2543Smrg#define SHASTEPX(stride) (-1) 81706f2543Smrg#define SHASTEPY(stride) -(stride) 82706f2543Smrg 83706f2543Smrg#else 84706f2543Smrg 85706f2543Smrg#define SCRLEFT(x,y,w,h) (x) 86706f2543Smrg#define SCRY(x,y,w,h) (y) 87706f2543Smrg#define SCRWIDTH(x,y,w,h) (w) 88706f2543Smrg#define FIRSTSHA(x,y,w,h) ((y) * shaStride + (x)) 89706f2543Smrg#define STEPDOWN(x,y,w,h) ((h)--) 90706f2543Smrg#define NEXTY(x,y,w,h) ((y)++) 91706f2543Smrg#define SHASTEPX(stride) (1) 92706f2543Smrg#define SHASTEPY(stride) (stride) 93706f2543Smrg 94706f2543Smrg#endif 95706f2543Smrg 96706f2543Smrgvoid 97706f2543SmrgFUNC (ScreenPtr pScreen, 98706f2543Smrg shadowBufPtr pBuf) 99706f2543Smrg{ 100706f2543Smrg RegionPtr damage = shadowDamage (pBuf); 101706f2543Smrg PixmapPtr pShadow = pBuf->pPixmap; 102706f2543Smrg int nbox = RegionNumRects (damage); 103706f2543Smrg BoxPtr pbox = RegionRects (damage); 104706f2543Smrg FbBits *shaBits; 105706f2543Smrg Data *shaBase, *shaLine, *sha; 106706f2543Smrg FbStride shaStride; 107706f2543Smrg int scrBase, scrLine, scr; 108706f2543Smrg int shaBpp; 109706f2543Smrg int shaXoff, shaYoff; /* XXX assumed to be zero */ 110706f2543Smrg int x, y, w, h, width; 111706f2543Smrg int i; 112706f2543Smrg Data *winBase = NULL, *win; 113706f2543Smrg CARD32 winSize; 114706f2543Smrg 115706f2543Smrg fbGetDrawable (&pShadow->drawable, shaBits, shaStride, shaBpp, shaXoff, shaYoff); 116706f2543Smrg shaBase = (Data *) shaBits; 117706f2543Smrg shaStride = shaStride * sizeof (FbBits) / sizeof (Data); 118706f2543Smrg#if (DANDEBUG > 1) 119706f2543Smrg 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); 120706f2543Smrg#endif 121706f2543Smrg while (nbox--) 122706f2543Smrg { 123706f2543Smrg x = pbox->x1; 124706f2543Smrg y = pbox->y1; 125706f2543Smrg w = (pbox->x2 - pbox->x1); 126706f2543Smrg h = pbox->y2 - pbox->y1; 127706f2543Smrg 128706f2543Smrg#if (DANDEBUG > 2) 129706f2543Smrg ErrorF (" |-> Redrawing box - Metrics: X=%d, Y=%d, Width=%d, Height=%d\n", x, y, w, h); 130706f2543Smrg#endif 131706f2543Smrg scrLine = SCRLEFT(x,y,w,h); 132706f2543Smrg shaLine = shaBase + FIRSTSHA(x,y,w,h); 133706f2543Smrg 134706f2543Smrg while (STEPDOWN(x,y,w,h)) 135706f2543Smrg { 136706f2543Smrg winSize = 0; 137706f2543Smrg scrBase = 0; 138706f2543Smrg width = SCRWIDTH(x,y,w,h); 139706f2543Smrg scr = scrLine; 140706f2543Smrg sha = shaLine; 141706f2543Smrg#if (DANDEBUG > 3) 142706f2543Smrg ErrorF (" | |-> StepDown - Metrics: width=%d, scr=%x, sha=%x\n", width, scr, sha); 143706f2543Smrg#endif 144706f2543Smrg while (width) 145706f2543Smrg { 146706f2543Smrg /* how much remains in this window */ 147706f2543Smrg i = scrBase + winSize - scr; 148706f2543Smrg if (i <= 0 || scr < scrBase) 149706f2543Smrg { 150706f2543Smrg winBase = (Data *) (*pBuf->window) (pScreen, 151706f2543Smrg SCRY(x,y,w,h), 152706f2543Smrg scr * sizeof (Data), 153706f2543Smrg SHADOW_WINDOW_WRITE, 154706f2543Smrg &winSize, 155706f2543Smrg pBuf->closure); 156706f2543Smrg if(!winBase) 157706f2543Smrg return; 158706f2543Smrg scrBase = scr; 159706f2543Smrg winSize /= sizeof (Data); 160706f2543Smrg i = winSize; 161706f2543Smrg#if(DANDEBUG > 4) 162706f2543Smrg 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); 163706f2543Smrg#endif 164706f2543Smrg } 165706f2543Smrg win = winBase + (scr - scrBase); 166706f2543Smrg if (i > width) 167706f2543Smrg i = width; 168706f2543Smrg width -= i; 169706f2543Smrg scr += i; 170706f2543Smrg#if(DANDEBUG > 5) 171706f2543Smrg ErrorF (" | | |-> Writing Line - Metrics: win=%x, sha=%x\n", win, sha); 172706f2543Smrg#endif 173706f2543Smrg while (i--) 174706f2543Smrg { 175706f2543Smrg#if(DANDEBUG > 6) 176706f2543Smrg ErrorF (" | | |-> Writing Pixel - Metrics: win=%x, sha=%d, remaining=%d\n", win, sha, i); 177706f2543Smrg#endif 178706f2543Smrg *win++ = *sha; 179706f2543Smrg sha += SHASTEPX(shaStride); 180706f2543Smrg } /* i */ 181706f2543Smrg } /* width */ 182706f2543Smrg shaLine += SHASTEPY(shaStride); 183706f2543Smrg NEXTY(x,y,w,h); 184706f2543Smrg } /* STEPDOWN */ 185706f2543Smrg pbox++; 186706f2543Smrg } /* nbox */ 187706f2543Smrg} 188