228 lines
7.2 KiB
C
228 lines
7.2 KiB
C
/*
|
|
*
|
|
* Copyright © 1998 Keith Packard
|
|
*
|
|
* Permission to use, copy, modify, distribute, and sell this software and its
|
|
* documentation for any purpose is hereby granted without fee, provided that
|
|
* the above copyright notice appear in all copies and that both that
|
|
* copyright notice and this permission notice appear in supporting
|
|
* documentation, and that the name of Keith Packard not be used in
|
|
* advertising or publicity pertaining to distribution of the software without
|
|
* specific, written prior permission. Keith Packard makes no
|
|
* representations about the suitability of this software for any purpose. It
|
|
* is provided "as is" without express or implied warranty.
|
|
*
|
|
* KEITH PACKARD DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
|
|
* INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
|
|
* EVENT SHALL KEITH PACKARD BE LIABLE FOR ANY SPECIAL, INDIRECT OR
|
|
* CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
|
|
* DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
|
|
* TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
|
* PERFORMANCE OF THIS SOFTWARE.
|
|
*/
|
|
|
|
#ifdef HAVE_DIX_CONFIG_H
|
|
#include <dix-config.h>
|
|
#endif
|
|
|
|
#include "fb.h"
|
|
#include <X11/fonts/fontstruct.h>
|
|
#include "dixfontstr.h"
|
|
|
|
static Bool
|
|
fbGlyphIn(RegionPtr pRegion, int x, int y, int width, int height)
|
|
{
|
|
BoxRec box;
|
|
BoxPtr pExtents = RegionExtents(pRegion);
|
|
|
|
/*
|
|
* Check extents by hand to avoid 16 bit overflows
|
|
*/
|
|
if (x < (int) pExtents->x1)
|
|
return FALSE;
|
|
if ((int) pExtents->x2 < x + width)
|
|
return FALSE;
|
|
if (y < (int) pExtents->y1)
|
|
return FALSE;
|
|
if ((int) pExtents->y2 < y + height)
|
|
return FALSE;
|
|
box.x1 = x;
|
|
box.x2 = x + width;
|
|
box.y1 = y;
|
|
box.y2 = y + height;
|
|
return RegionContainsRect(pRegion, &box) == rgnIN;
|
|
}
|
|
|
|
void
|
|
fbPolyGlyphBlt(DrawablePtr pDrawable,
|
|
GCPtr pGC,
|
|
int x,
|
|
int y,
|
|
unsigned int nglyph, CharInfoPtr * ppci, void *pglyphBase)
|
|
{
|
|
FbGCPrivPtr pPriv = fbGetGCPrivate(pGC);
|
|
CharInfoPtr pci;
|
|
unsigned char *pglyph; /* pointer bits in glyph */
|
|
int gx, gy;
|
|
int gWidth, gHeight; /* width and height of glyph */
|
|
FbStride gStride; /* stride of glyph */
|
|
void (*glyph) (FbBits *, FbStride, int, FbStip *, FbBits, int, int);
|
|
FbBits *dst = 0;
|
|
FbStride dstStride = 0;
|
|
int dstBpp = 0;
|
|
int dstXoff = 0, dstYoff = 0;
|
|
|
|
glyph = 0;
|
|
if (pGC->fillStyle == FillSolid && pPriv->and == 0) {
|
|
dstBpp = pDrawable->bitsPerPixel;
|
|
switch (dstBpp) {
|
|
case 8:
|
|
glyph = fbGlyph8;
|
|
break;
|
|
case 16:
|
|
glyph = fbGlyph16;
|
|
break;
|
|
case 32:
|
|
glyph = fbGlyph32;
|
|
break;
|
|
}
|
|
}
|
|
x += pDrawable->x;
|
|
y += pDrawable->y;
|
|
|
|
while (nglyph--) {
|
|
pci = *ppci++;
|
|
pglyph = FONTGLYPHBITS(pglyphBase, pci);
|
|
gWidth = GLYPHWIDTHPIXELS(pci);
|
|
gHeight = GLYPHHEIGHTPIXELS(pci);
|
|
if (gWidth && gHeight) {
|
|
gx = x + pci->metrics.leftSideBearing;
|
|
gy = y - pci->metrics.ascent;
|
|
if (glyph && gWidth <= sizeof(FbStip) * 8 &&
|
|
fbGlyphIn(fbGetCompositeClip(pGC), gx, gy, gWidth, gHeight)) {
|
|
fbGetDrawable(pDrawable, dst, dstStride, dstBpp, dstXoff,
|
|
dstYoff);
|
|
(*glyph) (dst + (gy + dstYoff) * dstStride, dstStride, dstBpp,
|
|
(FbStip *) pglyph, pPriv->xor, gx + dstXoff, gHeight);
|
|
fbFinishAccess(pDrawable);
|
|
}
|
|
else {
|
|
gStride = GLYPHWIDTHBYTESPADDED(pci) / sizeof(FbStip);
|
|
fbPushImage(pDrawable,
|
|
pGC,
|
|
(FbStip *) pglyph,
|
|
gStride, 0, gx, gy, gWidth, gHeight);
|
|
}
|
|
}
|
|
x += pci->metrics.characterWidth;
|
|
}
|
|
}
|
|
|
|
void
|
|
fbImageGlyphBlt(DrawablePtr pDrawable,
|
|
GCPtr pGC,
|
|
int x,
|
|
int y,
|
|
unsigned int nglyph, CharInfoPtr * ppciInit, void *pglyphBase)
|
|
{
|
|
FbGCPrivPtr pPriv = fbGetGCPrivate(pGC);
|
|
CharInfoPtr *ppci;
|
|
CharInfoPtr pci;
|
|
unsigned char *pglyph; /* pointer bits in glyph */
|
|
int gWidth, gHeight; /* width and height of glyph */
|
|
FbStride gStride; /* stride of glyph */
|
|
Bool opaque;
|
|
int n;
|
|
int gx, gy;
|
|
void (*glyph) (FbBits *, FbStride, int, FbStip *, FbBits, int, int);
|
|
FbBits *dst = 0;
|
|
FbStride dstStride = 0;
|
|
int dstBpp = 0;
|
|
int dstXoff = 0, dstYoff = 0;
|
|
|
|
glyph = 0;
|
|
if (pPriv->and == 0) {
|
|
dstBpp = pDrawable->bitsPerPixel;
|
|
switch (dstBpp) {
|
|
case 8:
|
|
glyph = fbGlyph8;
|
|
break;
|
|
case 16:
|
|
glyph = fbGlyph16;
|
|
break;
|
|
case 32:
|
|
glyph = fbGlyph32;
|
|
break;
|
|
}
|
|
}
|
|
|
|
x += pDrawable->x;
|
|
y += pDrawable->y;
|
|
|
|
if (TERMINALFONT(pGC->font)
|
|
&& !glyph) {
|
|
opaque = TRUE;
|
|
}
|
|
else {
|
|
int xBack, widthBack;
|
|
int yBack, heightBack;
|
|
|
|
ppci = ppciInit;
|
|
n = nglyph;
|
|
widthBack = 0;
|
|
while (n--)
|
|
widthBack += (*ppci++)->metrics.characterWidth;
|
|
|
|
xBack = x;
|
|
if (widthBack < 0) {
|
|
xBack += widthBack;
|
|
widthBack = -widthBack;
|
|
}
|
|
yBack = y - FONTASCENT(pGC->font);
|
|
heightBack = FONTASCENT(pGC->font) + FONTDESCENT(pGC->font);
|
|
fbSolidBoxClipped(pDrawable,
|
|
fbGetCompositeClip(pGC),
|
|
xBack,
|
|
yBack,
|
|
xBack + widthBack,
|
|
yBack + heightBack,
|
|
fbAnd(GXcopy, pPriv->bg, pPriv->pm),
|
|
fbXor(GXcopy, pPriv->bg, pPriv->pm));
|
|
opaque = FALSE;
|
|
}
|
|
|
|
ppci = ppciInit;
|
|
while (nglyph--) {
|
|
pci = *ppci++;
|
|
pglyph = FONTGLYPHBITS(pglyphBase, pci);
|
|
gWidth = GLYPHWIDTHPIXELS(pci);
|
|
gHeight = GLYPHHEIGHTPIXELS(pci);
|
|
if (gWidth && gHeight) {
|
|
gx = x + pci->metrics.leftSideBearing;
|
|
gy = y - pci->metrics.ascent;
|
|
if (glyph && gWidth <= sizeof(FbStip) * 8 &&
|
|
fbGlyphIn(fbGetCompositeClip(pGC), gx, gy, gWidth, gHeight)) {
|
|
fbGetDrawable(pDrawable, dst, dstStride, dstBpp, dstXoff,
|
|
dstYoff);
|
|
(*glyph) (dst + (gy + dstYoff) * dstStride, dstStride, dstBpp,
|
|
(FbStip *) pglyph, pPriv->fg, gx + dstXoff, gHeight);
|
|
fbFinishAccess(pDrawable);
|
|
}
|
|
else {
|
|
gStride = GLYPHWIDTHBYTESPADDED(pci) / sizeof(FbStip);
|
|
fbPutXYImage(pDrawable,
|
|
fbGetCompositeClip(pGC),
|
|
pPriv->fg,
|
|
pPriv->bg,
|
|
pPriv->pm,
|
|
GXcopy,
|
|
opaque,
|
|
gx,
|
|
gy,
|
|
gWidth, gHeight, (FbStip *) pglyph, gStride, 0);
|
|
}
|
|
}
|
|
x += pci->metrics.characterWidth;
|
|
}
|
|
}
|