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