fontink.c revision 23a0898a
123a0898aSmrg/* $Xorg: fontink.c,v 1.4 2001/02/09 02:04:02 xorgcvs Exp $ */ 223a0898aSmrg 323a0898aSmrg/* 423a0898aSmrg 523a0898aSmrgCopyright 1990, 1998 The Open Group 623a0898aSmrg 723a0898aSmrgPermission to use, copy, modify, distribute, and sell this software and its 823a0898aSmrgdocumentation for any purpose is hereby granted without fee, provided that 923a0898aSmrgthe above copyright notice appear in all copies and that both that 1023a0898aSmrgcopyright notice and this permission notice appear in supporting 1123a0898aSmrgdocumentation. 1223a0898aSmrg 1323a0898aSmrgThe above copyright notice and this permission notice shall be included 1423a0898aSmrgin all copies or substantial portions of the Software. 1523a0898aSmrg 1623a0898aSmrgTHE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS 1723a0898aSmrgOR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF 1823a0898aSmrgMERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. 1923a0898aSmrgIN NO EVENT SHALL THE OPEN GROUP BE LIABLE FOR ANY CLAIM, DAMAGES OR 2023a0898aSmrgOTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, 2123a0898aSmrgARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR 2223a0898aSmrgOTHER DEALINGS IN THE SOFTWARE. 2323a0898aSmrg 2423a0898aSmrgExcept as contained in this notice, the name of The Open Group shall 2523a0898aSmrgnot be used in advertising or otherwise to promote the sale, use or 2623a0898aSmrgother dealings in this Software without prior written authorization 2723a0898aSmrgfrom The Open Group. 2823a0898aSmrg 2923a0898aSmrg*/ 3023a0898aSmrg/* $XFree86: xc/lib/font/bitmap/fontink.c,v 1.6 2001/01/17 19:43:27 dawes Exp $ */ 3123a0898aSmrg 3223a0898aSmrg/* 3323a0898aSmrg * Author: Keith Packard, MIT X Consortium 3423a0898aSmrg */ 3523a0898aSmrg 3623a0898aSmrg#ifdef HAVE_CONFIG_H 3723a0898aSmrg#include <config.h> 3823a0898aSmrg#endif 3923a0898aSmrg 4023a0898aSmrg#include <X11/fonts/fntfilst.h> 4123a0898aSmrg#include <X11/fonts/bitmap.h> 4223a0898aSmrg#include <X11/fonts/bdfint.h> 4323a0898aSmrg 4423a0898aSmrgstatic unsigned char ink_mask_msb[8] = { 4523a0898aSmrg 0x80, 0x40, 0x20, 0x10, 0x08, 0x04, 0x02, 0x01, 4623a0898aSmrg}; 4723a0898aSmrg 4823a0898aSmrgstatic unsigned char ink_mask_lsb[8] = { 4923a0898aSmrg 0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80, 5023a0898aSmrg}; 5123a0898aSmrg 5223a0898aSmrgvoid 5323a0898aSmrgFontCharInkMetrics(FontPtr pFont, CharInfoPtr pCI, xCharInfo *pInk) 5423a0898aSmrg{ 5523a0898aSmrg int leftBearing, 5623a0898aSmrg ascent, 5723a0898aSmrg descent; 5823a0898aSmrg register int vpos, 5923a0898aSmrg hpos, 6023a0898aSmrg bpos = 0; 6123a0898aSmrg int bitmapByteWidth, 6223a0898aSmrg bitmapByteWidthPadded; 6323a0898aSmrg int bitmapBitWidth; 6423a0898aSmrg int span; 6523a0898aSmrg register unsigned char *p; 6623a0898aSmrg unsigned char *ink_mask = 0; 6723a0898aSmrg register int bmax; 6823a0898aSmrg register unsigned char charbits; 6923a0898aSmrg 7023a0898aSmrg if (pFont->bit == MSBFirst) 7123a0898aSmrg ink_mask = ink_mask_msb; 7223a0898aSmrg else if (pFont->bit == LSBFirst) 7323a0898aSmrg ink_mask = ink_mask_lsb; 7423a0898aSmrg pInk->characterWidth = pCI->metrics.characterWidth; 7523a0898aSmrg pInk->attributes = pCI->metrics.attributes; 7623a0898aSmrg 7723a0898aSmrg leftBearing = pCI->metrics.leftSideBearing; 7823a0898aSmrg ascent = pCI->metrics.ascent; 7923a0898aSmrg descent = pCI->metrics.descent; 8023a0898aSmrg bitmapBitWidth = GLYPHWIDTHPIXELS(pCI); 8123a0898aSmrg bitmapByteWidth = GLYPHWIDTHBYTES(pCI); 8223a0898aSmrg bitmapByteWidthPadded = BYTES_PER_ROW(bitmapBitWidth, pFont->glyph); 8323a0898aSmrg span = bitmapByteWidthPadded - bitmapByteWidth; 8423a0898aSmrg 8523a0898aSmrg p = (unsigned char *) pCI->bits; 8623a0898aSmrg for (vpos = descent + ascent; --vpos >= 0;) { 8723a0898aSmrg for (hpos = bitmapByteWidth; --hpos >= 0;) { 8823a0898aSmrg if (*p++ != 0) 8923a0898aSmrg goto found_ascent; 9023a0898aSmrg } 9123a0898aSmrg p += span; 9223a0898aSmrg } 9323a0898aSmrg /* 9423a0898aSmrg * special case -- font with no bits gets all zeros 9523a0898aSmrg */ 9623a0898aSmrg pInk->leftSideBearing = leftBearing; 9723a0898aSmrg pInk->rightSideBearing = leftBearing; 9823a0898aSmrg pInk->ascent = 0; 9923a0898aSmrg pInk->descent = 0; 10023a0898aSmrg return; 10123a0898aSmrgfound_ascent: 10223a0898aSmrg pInk->ascent = vpos - descent + 1; 10323a0898aSmrg 10423a0898aSmrg p = ((unsigned char *) pCI->bits) + bitmapByteWidthPadded * 10523a0898aSmrg (descent + ascent - 1) + bitmapByteWidth; 10623a0898aSmrg 10723a0898aSmrg for (vpos = descent + ascent; --vpos >= 0;) { 10823a0898aSmrg for (hpos = bitmapByteWidth; --hpos >= 0;) { 10923a0898aSmrg if (*--p != 0) 11023a0898aSmrg goto found_descent; 11123a0898aSmrg } 11223a0898aSmrg p -= span; 11323a0898aSmrg } 11423a0898aSmrgfound_descent: 11523a0898aSmrg pInk->descent = vpos - ascent + 1; 11623a0898aSmrg 11723a0898aSmrg bmax = 8; 11823a0898aSmrg for (hpos = 0; hpos < bitmapByteWidth; hpos++) { 11923a0898aSmrg charbits = 0; 12023a0898aSmrg p = (unsigned char *) pCI->bits + hpos; 12123a0898aSmrg for (vpos = descent + ascent; --vpos >= 0; p += bitmapByteWidthPadded) 12223a0898aSmrg charbits |= *p; 12323a0898aSmrg if (charbits) { 12423a0898aSmrg if (hpos == bitmapByteWidth - 1) 12523a0898aSmrg bmax = bitmapBitWidth - (hpos << 3); 12623a0898aSmrg p = ink_mask; 12723a0898aSmrg for (bpos = bmax; --bpos >= 0;) { 12823a0898aSmrg if (charbits & *p++) 12923a0898aSmrg goto found_left; 13023a0898aSmrg } 13123a0898aSmrg } 13223a0898aSmrg } 13323a0898aSmrgfound_left: 13423a0898aSmrg pInk->leftSideBearing = leftBearing + (hpos << 3) + bmax - bpos - 1; 13523a0898aSmrg 13623a0898aSmrg bmax = bitmapBitWidth - ((bitmapByteWidth - 1) << 3); 13723a0898aSmrg for (hpos = bitmapByteWidth; --hpos >= 0;) { 13823a0898aSmrg charbits = 0; 13923a0898aSmrg p = (unsigned char *) pCI->bits + hpos; 14023a0898aSmrg for (vpos = descent + ascent; --vpos >= 0; p += bitmapByteWidthPadded) 14123a0898aSmrg charbits |= *p; 14223a0898aSmrg if (charbits) { 14323a0898aSmrg p = ink_mask + bmax; 14423a0898aSmrg for (bpos = bmax; --bpos >= 0;) { 14523a0898aSmrg if (charbits & *--p) 14623a0898aSmrg goto found_right; 14723a0898aSmrg } 14823a0898aSmrg } 14923a0898aSmrg bmax = 8; 15023a0898aSmrg } 15123a0898aSmrgfound_right: 15223a0898aSmrg pInk->rightSideBearing = leftBearing + (hpos << 3) + bpos + 1; 15323a0898aSmrg} 15423a0898aSmrg 15523a0898aSmrg#define ISBITONMSB(x, line) ((line)[(x)/8] & (1 << (7-((x)%8)))) 15623a0898aSmrg#define SETBITMSB(x, line) ((line)[(x)/8] |= (1 << (7-((x)%8)))) 15723a0898aSmrg#define ISBITONLSB(x, line) ((line)[(x)/8] & (1 << ((x)%8))) 15823a0898aSmrg#define SETBITLSB(x, line) ((line)[(x)/8] |= (1 << ((x)%8))) 15923a0898aSmrg 16023a0898aSmrg#define Min(a,b) ((a)<(b)?(a):(b)) 16123a0898aSmrg#define Max(a,b) ((a)>(b)?(a):(b)) 16223a0898aSmrg 16323a0898aSmrgvoid 16423a0898aSmrgFontCharReshape(FontPtr pFont, CharInfoPtr pSrc, CharInfoPtr pDst) 16523a0898aSmrg{ 16623a0898aSmrg int x, 16723a0898aSmrg y; 16823a0898aSmrg unsigned char *in_line, 16923a0898aSmrg *out_line; 17023a0898aSmrg unsigned char *oldglyph, 17123a0898aSmrg *newglyph; 17223a0898aSmrg int inwidth; 17323a0898aSmrg int outwidth, 17423a0898aSmrg outheight; 17523a0898aSmrg int out_bytes, 17623a0898aSmrg in_bytes; 17723a0898aSmrg int y_min, 17823a0898aSmrg y_max, 17923a0898aSmrg x_min, 18023a0898aSmrg x_max; 18123a0898aSmrg 18223a0898aSmrg newglyph = (unsigned char *) pDst->bits; 18323a0898aSmrg outwidth = pDst->metrics.rightSideBearing - pDst->metrics.leftSideBearing; 18423a0898aSmrg outheight = pDst->metrics.descent + pDst->metrics.ascent; 18523a0898aSmrg out_bytes = BYTES_PER_ROW(outwidth, pFont->glyph); 18623a0898aSmrg 18723a0898aSmrg oldglyph = (unsigned char *) pSrc->bits; 18823a0898aSmrg inwidth = pSrc->metrics.rightSideBearing - pSrc->metrics.leftSideBearing; 18923a0898aSmrg in_bytes = BYTES_PER_ROW(inwidth, pFont->glyph); 19023a0898aSmrg 19123a0898aSmrg bzero(newglyph, out_bytes * outheight); 19223a0898aSmrg in_line = oldglyph; 19323a0898aSmrg out_line = newglyph; 19423a0898aSmrg y_min = Max(-pSrc->metrics.ascent, -pDst->metrics.ascent); 19523a0898aSmrg y_max = Min(pSrc->metrics.descent, pDst->metrics.descent); 19623a0898aSmrg x_min = Max(pSrc->metrics.leftSideBearing, pDst->metrics.leftSideBearing); 19723a0898aSmrg x_max = Min(pSrc->metrics.rightSideBearing, pDst->metrics.rightSideBearing); 19823a0898aSmrg in_line += (y_min + pSrc->metrics.ascent) * in_bytes; 19923a0898aSmrg out_line += (y_min + pDst->metrics.ascent) * out_bytes; 20023a0898aSmrg if (pFont->bit == MSBFirst) { 20123a0898aSmrg for (y = y_min; y < y_max; y++) { 20223a0898aSmrg for (x = x_min; x < x_max; x++) { 20323a0898aSmrg if (ISBITONMSB(x - pSrc->metrics.leftSideBearing, in_line)) 20423a0898aSmrg SETBITMSB(x - pDst->metrics.leftSideBearing, out_line); 20523a0898aSmrg } 20623a0898aSmrg in_line += in_bytes; 20723a0898aSmrg out_line += out_bytes; 20823a0898aSmrg } 20923a0898aSmrg } else { 21023a0898aSmrg for (y = y_min; y < y_max; y++) { 21123a0898aSmrg for (x = x_min; x < x_max; x++) { 21223a0898aSmrg if (ISBITONLSB(x - pSrc->metrics.leftSideBearing, in_line)) 21323a0898aSmrg SETBITLSB(x - pDst->metrics.leftSideBearing, out_line); 21423a0898aSmrg } 21523a0898aSmrg in_line += in_bytes; 21623a0898aSmrg out_line += out_bytes; 21723a0898aSmrg } 21823a0898aSmrg } 21923a0898aSmrg} 220