📄 cfbglblt8.c
字号:
/* * Poly glyph blt for 8 bit displays. Accepts * an arbitrary font <= 32 bits wide, in Copy mode only. *//*Copyright 1989 by the Massachusetts Institute of TechnologyPermission to use, copy, modify, and distribute this software and itsdocumentation for any purpose and without fee is hereby granted,provided that the above copyright notice appear in all copies and thatboth that copyright notice and this permission notice appear insupporting documentation, and that the name of M.I.T. not be used inadvertising or publicity pertaining to distribution of the softwarewithout specific, written prior permission. M.I.T. makes norepresentations about the suitability of this software for anypurpose. It is provided "as is" without express or implied warranty.*//* $XConsortium: cfbglblt8.c,v 5.23 91/07/18 23:44:50 keith Exp $ */#include "X.h"#include "Xmd.h"#include "Xproto.h"#include "cfb.h"#include "fontstruct.h"#include "dixfontstr.h"#include "gcstruct.h"#include "windowstr.h"#include "scrnintstr.h"#include "pixmapstr.h"#include "regionstr.h"#include "cfbmskbits.h"#include "cfb8bit.h"#define BOX_OVERLAP(box1, box2, xoffset, yoffset) \ ((box1)->x1 <= ((int) (box2)->x2 + (xoffset)) && \ ((int) (box2)->x1 + (xoffset)) <= (box1)->x2 && \ (box1)->y1 <= ((int) (box2)->y2 + (yoffset)) && \ ((int) (box2)->y1 + (yoffset)) <= (box1)->y2)#define BOX_CONTAINS(box1, box2, xoffset, yoffset) \ ((box1)->x1 <= ((int) (box2)->x1 + (xoffset)) && \ ((int) (box2)->x2 + (xoffset)) <= (box1)->x2 && \ (box1)->y1 <= ((int) (box2)->y1 + (yoffset)) && \ ((int) (box2)->y2 + (yoffset)) <= (box1)->y2)#if PPW == 4#if GLYPHPADBYTES != 4#define USE_LEFTBITS#endif#ifdef USE_LEFTBITStypedef unsigned char *glyphPointer;extern long endtab[];#define GlyphBits(bits,width,dst) getleftbits(bits,width,dst); \ (dst) &= widthMask; \ (bits) += widthGlyph;#define GlyphBitsS(bits,width,dst,off) GlyphBits(bits,width,dst); \ dst = BitRight (dst, off);#elsetypedef unsigned long *glyphPointer;#define GlyphBits(bits,width,dst) dst = *bits++;#define GlyphBitsS(bits,width,dst,off) dst = BitRight(*bits++, off);#endif#ifdef GLYPHROP#define cfbPolyGlyphBlt8 cfbPolyGlyphRop8#define cfbPolyGlyphBlt8Clipped cfbPolyGlyphRop8Clipped#undef WriteFourBits#define WriteFourBits(dst,pixel,bits) RRopFourBits(dst,bits)#endifstatic void cfbPolyGlyphBlt8Clipped();#if defined(HAS_STIPPLE_CODE) && !defined(GLYPHROP) && !defined(USE_LEFTBITS)#define USE_STIPPLE_CODE#endif#if defined(__GNUC__) && !defined(GLYPHROP) && (defined(mc68020) || defined(mc68000) || defined(__mc68000__)) && !defined(USE_LEFTBITS)#include <stip68kgnu.h>#endifvoidcfbPolyGlyphBlt8 (pDrawable, pGC, x, y, nglyph, ppci, pglyphBase) DrawablePtr pDrawable; GCPtr pGC; int x, y; unsigned int nglyph; CharInfoPtr *ppci; /* array of character info */ unsigned char *pglyphBase; /* start of array of glyphs */{ register unsigned long c;#ifndef GLYPHROP register unsigned long pixel;#endif register unsigned long *dst; register glyphPointer glyphBits; register int xoff; register int ewTmp; FontPtr pfont = pGC->font; CharInfoPtr pci; unsigned long *dstLine; unsigned long *pdstBase; int hTmp; int bwidthDst; int widthDst; int h; int ew; BoxRec bbox; /* for clipping */ int widthDiff; int w; RegionPtr clip; BoxPtr extents;#ifdef USE_LEFTBITS int widthGlyph; unsigned long widthMask;#endif#ifndef STIPPLE#ifdef USE_STIPPLE_CODE void (*stipple)(); extern void stipplestack (), stipplestackte (); stipple = stipplestack; if (FONTCONSTMETRICS(pfont)) stipple = stipplestackte;#endif#endif x += pDrawable->x; y += pDrawable->y; /* compute an approximate (but covering) bounding box */ bbox.x1 = 0; if ((ppci[0]->metrics.leftSideBearing < 0)) bbox.x1 = ppci[0]->metrics.leftSideBearing; h = nglyph - 1; w = ppci[h]->metrics.rightSideBearing; while (--h >= 0) w += ppci[h]->metrics.characterWidth; bbox.x2 = w; bbox.y1 = -FONTMAXBOUNDS(pfont,ascent); bbox.y2 = FONTMAXBOUNDS(pfont,descent); clip = ((cfbPrivGC *)(pGC->devPrivates[cfbGCPrivateIndex].ptr))->pCompositeClip; extents = &clip->extents; if (!clip->data) { if (!BOX_CONTAINS(extents, &bbox, x, y)) { if (BOX_OVERLAP (extents, &bbox, x, y)) cfbPolyGlyphBlt8Clipped(pDrawable, pGC, x, y, nglyph, ppci, pglyphBase); return; } } else { /* check to make sure some of the text appears on the screen */ if (!BOX_OVERLAP (extents, &bbox, x, y)) return; bbox.x1 += x; bbox.x2 += x; bbox.y1 += y; bbox.y2 += y; switch ((*pGC->pScreen->RectIn)(clip, &bbox)) { case rgnPART: cfbPolyGlyphBlt8Clipped(pDrawable, pGC, x, y, nglyph, ppci, pglyphBase); case rgnOUT: return; } }#ifdef GLYPHROP cfb8CheckStipple (pGC->alu, pGC->fgPixel, pGC->planemask);#else pixel = ((cfbPrivGCPtr) pGC->devPrivates[cfbGCPrivateIndex].ptr)->xor;#endif cfbGetTypedWidthAndPointer (pDrawable, bwidthDst, pdstBase, char, unsigned long) widthDst = bwidthDst >> 2; SET_REGISTERS_FOR_WRITING(pDrawable->pScreen, ~0, GXcopy); while (nglyph--) { pci = *ppci++; glyphBits = (glyphPointer) FONTGLYPHBITS(pglyphBase,pci); xoff = x + pci->metrics.leftSideBearing; dstLine = pdstBase + (y - pci->metrics.ascent) * widthDst + (xoff >> 2); x += pci->metrics.characterWidth; if (hTmp = pci->metrics.descent + pci->metrics.ascent) { xoff &= 0x3;#ifdef STIPPLE STIPPLE(dstLine,glyphBits,pixel,bwidthDst,hTmp,xoff);#else#ifdef USE_STIPPLE_CODE (*stipple)(dstLine,glyphBits,pixel,bwidthDst,hTmp,xoff);#else#ifdef USE_LEFTBITS w = pci->metrics.rightSideBearing - pci->metrics.leftSideBearing; widthGlyph = PADGLYPHWIDTHBYTES(w); widthMask = endtab[w];#endif do { dst = dstLine; dstLine = (unsigned long *) (((char *) dstLine) + bwidthDst); GlyphBits(glyphBits, w, c) WriteFourBits(dst, pixel, GetFourBits(BitRight(c,xoff))); dst++; c = BitLeft(c,4-xoff); while (c) { WriteFourBits(dst, pixel, GetFourBits(c)); NextFourBits(c); dst++; } } while (--hTmp);#endif /* USE_STIPPLE_CODE else */#endif /* STIPPLE else */ } }}static voidcfbPolyGlyphBlt8Clipped (pDrawable, pGC, x, y, nglyph, ppci, pglyphBase) DrawablePtr pDrawable; GCPtr pGC; int x, y; unsigned int nglyph; CharInfoPtr *ppci; /* array of character info */ unsigned char *pglyphBase; /* start of array of glyphs */{ register unsigned long c;#ifndef GLYPHROP register unsigned long pixel;#endif register unsigned long *dst; register glyphPointer glyphBits; register int xoff; register int ewTmp; unsigned long c1; CharInfoPtr pci; FontPtr pfont = pGC->font; unsigned long *dstLine; unsigned long *pdstBase; unsigned long *cTmp, *clips; int maxAscent, maxDescent; int minLeftBearing; int hTmp; int widthDst; int bwidthDst; int ew; int xG, yG; BoxPtr pBox; int numRects; int widthDiff; int w; RegionPtr pRegion; int yBand;#ifdef USE_LEFTBITS int widthGlyph; unsigned long widthMask;#endif#ifdef GLYPHROP cfb8CheckStipple (pGC->alu, pGC->fgPixel, pGC->planemask);#else pixel = ((cfbPrivGCPtr) pGC->devPrivates[cfbGCPrivateIndex].ptr)->xor;#endif cfbGetTypedWidthAndPointer (pDrawable, bwidthDst, pdstBase, char, unsigned long) widthDst = bwidthDst >> 2; maxAscent = FONTMAXBOUNDS(pfont,ascent); maxDescent = FONTMAXBOUNDS(pfont,descent); minLeftBearing = FONTMINBOUNDS(pfont,leftSideBearing); SET_REGISTERS_FOR_WRITING(pDrawable->pScreen, ~0, GXcopy); pRegion = ((cfbPrivGC *)(pGC->devPrivates[cfbGCPrivateIndex].ptr))->pCompositeClip; pBox = REGION_RECTS(pRegion); numRects = REGION_NUM_RECTS (pRegion); while (numRects && pBox->y2 <= y - maxAscent) { ++pBox; --numRects; } if (!numRects || pBox->y1 >= y + maxDescent) return; yBand = pBox->y1; while (numRects && pBox->y1 == yBand && pBox->x2 <= x + minLeftBearing) { ++pBox; --numRects; } if (!numRects) return; clips = (unsigned long *)ALLOCATE_LOCAL ((maxAscent + maxDescent) * sizeof (unsigned long)); while (nglyph--) { pci = *ppci++; glyphBits = (glyphPointer) FONTGLYPHBITS(pglyphBase,pci); w = pci->metrics.rightSideBearing - pci->metrics.leftSideBearing; xG = x + pci->metrics.leftSideBearing; yG = y - pci->metrics.ascent; x += pci->metrics.characterWidth; if (hTmp = pci->metrics.descent + pci->metrics.ascent) { dstLine = pdstBase + yG * widthDst + (xG >> 2); xoff = xG & 0x3;#ifdef USE_LEFTBITS w = pci->metrics.rightSideBearing - pci->metrics.leftSideBearing; widthGlyph = PADGLYPHWIDTHBYTES(w); widthMask = endtab[w];#endif switch (cfb8ComputeClipMasks32 (pBox, numRects, xG, yG, w, hTmp, clips)) { case rgnPART:#ifdef USE_LEFTBITS cTmp = clips; do { dst = dstLine; dstLine = (unsigned long *) (((char *) dstLine) + bwidthDst); GlyphBits(glyphBits, w, c) c &= *cTmp++; if (c) { WriteFourBits(dst, pixel, GetFourBits(BitRight(c,xoff))); c = BitLeft(c,4 - xoff); dst++; while (c) { WriteFourBits(dst, pixel, GetFourBits(c)); NextFourBits(c); dst++; } } } while (--hTmp); break;#else { int h; h = hTmp; do { --h; clips[h] = clips[h] & glyphBits[h]; } while (h); } glyphBits = clips; /* fall through */#endif case rgnIN:#ifdef STIPPLE STIPPLE(dstLine,glyphBits,pixel,bwidthDst,hTmp,xoff);#else#ifdef USE_STIPPLE_CODE stipplestackte(dstLine,glyphBits,pixel,bwidthDst,hTmp,xoff);#else do { dst = dstLine; dstLine = (unsigned long *) (((char *) dstLine) + bwidthDst); GlyphBits(glyphBits, w, c) if (c) { WriteFourBits(dst, pixel, GetFourBits(BitRight(c,xoff))); c = BitLeft(c,4-xoff); dst++; while (c) { WriteFourBits(dst, pixel, GetFourBits(c)); NextFourBits(c); dst++; } } } while (--hTmp);#endif /* USE_STIPPLE_CODE else */#endif /* STIPPLE else */ break; } } } DEALLOCATE_LOCAL (clips);}#endif /* PPW == 4 */
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -