/* Copyright (c) 1997 by Mark Leisher Copyright (c) 1998-2001 by Juliusz Chroboczek Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ /* $XFree86: xc/lib/font/FreeType/ftfuncs.c,v 1.20 2001/09/04 13:49:15 dawes Exp $ */ #ifndef FONT_MODULE #include #else #include "xf86_ansic.h" #endif #include "fontmisc.h" #include "fntfilst.h" #include "fontutil.h" #include "FSproto.h" #include "freetype.h" #include "ttconfig.h" #include "fontenc.h" #include "ft.h" #include "ftfuncs.h" /* The propery names for all the XLFD properties. */ static char *xlfd_props[] = { "FOUNDRY", "FAMILY_NAME", "WEIGHT_NAME", "SLANT", "SETWIDTH_NAME", "ADD_STYLE_NAME", "PIXEL_SIZE", "POINT_SIZE", "RESOLUTION_X", "RESOLUTION_Y", "SPACING", "AVERAGE_WIDTH", "CHARSET_REGISTRY", "CHARSET_ENCODING", }; static int ftypeInitP = 0; /* is the engine initialised? */ static TT_Engine ftypeEngine; /* the engine */ static TTFFace *faceTable[NUMFACEBUCKETS]; static unsigned hash(char *string) { int i; unsigned u=0; for(i=0; string[i]!='\0'; i++) u=(u<<2)+(unsigned char)string[i]; return u; } static int ifloor(int x, int y) { if(x>=0) return x/y; else return x/y-1; } static int iceil(int x, int y) { return ifloor(x+y-1, y); } static int FreeTypeOpenFace(TTFFace **facep, char *fileName) { int ftrc; int bucket; TTFFace *face, *otherFace; char *realFileName; int faceNumber; /* Make sure FreeType is initialized. */ if (!ftypeInitP) { /* Report initialization errors as an allocation error because X does * not provide any other kind of error code related to intialization. */ if (ftrc=TT_Init_FreeType(&ftypeEngine)) { MUMBLE1("Error initializing ftypeEngine: %d\n", ftrc); return AllocError; } ftypeInitP = 1; } /* Try to find a matching face in the hashtable */ bucket=hash(fileName)%NUMFACEBUCKETS; otherFace=faceTable[bucket]; while(otherFace) { if(!strcmp(otherFace->filename,fileName)) break; otherFace=otherFace->next; } if(otherFace) { MUMBLE1("Returning cached face: %s\n", otherFace->filename); *facep=otherFace; return Successful; } /* No cached match; need to make a new one */ if((face=(TTFFace*)xalloc(sizeof(TTFFace)))==NULL) { MUMBLE("Couldn't allocate TTFFace\n"); return AllocError; } if((face->filename=(char*)xalloc(strlen(fileName)+1))==NULL) { xfree(face); return AllocError; } strcpy(face->filename, fileName); face->instances=0; if(ttf_checkForTTCName(fileName, &realFileName, &faceNumber)) { ftrc=TT_Open_Collection(ftypeEngine, realFileName, faceNumber, &face->face); xfree(realFileName); } else { ftrc=TT_Open_Face(ftypeEngine, fileName, &face->face); } if(ftrc) { MUMBLE1("Couldn't open face: %d\n", ftrc); xfree(face->filename); xfree(face); return BadFontPath; } if(ftrc=TT_Get_Face_Properties(face->face, &face->properties)) { MUMBLE1("Couldn't get properties: %d\n", ftrc); TT_Close_Face(face->face); xfree(face->filename); xfree(face); return BadFontPath; } if(ftrc=TT_New_Glyph(face->face, &face->glyph)) { MUMBLE1("Couldn't allocate glyph container: %d\n", ftrc); TT_Close_Face(face->face); xfree(face->filename); xfree(face); return BadFontPath; } /* Insert face in hashtable and return it */ face->next=faceTable[bucket]; faceTable[bucket]=face; *facep=face; return Successful; } static void FreeTypeFreeFace(TTFFace *face) { int bucket; TTFFace *otherFace; if(!face->instances) { bucket=hash(face->filename)%NUMFACEBUCKETS; if(faceTable[bucket]==face) faceTable[bucket]=face->next; else { otherFace=faceTable[bucket]; while(otherFace) { if(otherFace->next==face) break; otherFace=otherFace->next; } if(otherFace && otherFace->next) otherFace->next=otherFace->next->next; else ErrorF("Congratulations, \ you've found a bug in the TrueType backend: freeing unknown face\n"); } MUMBLE1("Closing face: %s\n", face->filename); TT_Close_Face(face->face); xfree(face->filename); xfree(face); } } static int TransEqual(TTFNormalisedTransformation *t1, TTFNormalisedTransformation *t2) { if(t1->scale!=t2->scale) return 0; else if(t1->xres!=t2->xres || t1->yres!=t2->yres) return 0; else if(t1->nonIdentity != t2->nonIdentity) return 0; else if(t1->nonIdentity && t2->nonIdentity) { return t1->matrix.xx == t2->matrix.xx && t1->matrix.yx == t2->matrix.yx && t1->matrix.yy == t2->matrix.yy && t1->matrix.xy == t2->matrix.xy; } else return 1; } static int BitmapFormatEqual(FontBitmapFormat *f1, FontBitmapFormat *f2) { return f1->bit==f2->bit && f1->byte==f2->byte && f1->glyph==f2->glyph; } static int TTFInstanceMatch(TTFInstance *instance, char *fileName, TTFNormalisedTransformation *trans, int charcell, FontBitmapFormat *bmfmt) { if(strcmp(instance->face->filename, fileName)) { return 0; } else if(!TransEqual(&instance->transformation, trans)) { return 0; } else if((charcell && instance->monospaced!=2) || (!charcell && instance->monospaced==2)) { return 0; } else if(!BitmapFormatEqual(&instance->bmfmt, bmfmt)) { return 0; } else { return 1; } } static int FreeTypeOpenInstance(TTFInstance **instancep, char *fileName, TTFNormalisedTransformation *trans, int charcell, FontBitmapFormat *bmfmt) { int ftrc, xrc; TTFInstance *instance, *otherInstance; TT_Face_Properties *properties; TTFFace *face; xrc=FreeTypeOpenFace(&face, fileName); if(xrc!=Successful) { return xrc; } /* Search for a matching instance */ for(otherInstance=face->instances; otherInstance; otherInstance=otherInstance->next) { if(TTFInstanceMatch(otherInstance, fileName, trans, charcell, bmfmt)) break; } if(otherInstance) { MUMBLE("Returning cached instance\n"); otherInstance->refcount++; *instancep=otherInstance; return Successful; } /* None matching found */ if((instance=(TTFInstance*)xalloc(sizeof(TTFInstance)))==0) { MUMBLE("Unable to allocate TTFInstance\n"); return AllocError; } instance->refcount=1; instance->face=face; properties=&instance->face->properties; instance->monospaced=charcell?2:0; instance->charcellMetrics=0; instance->width=0; instance->transformation=*trans; instance->bmfmt=*bmfmt; instance->nglyphs=properties->num_Glyphs; instance->glyphs=0; instance->available=0; if(ftrc=TT_New_Instance(instance->face->face, &instance->instance)) { MUMBLE("Couldn't create instance\n"); FreeTypeFreeFace(instance->face); xfree(instance); return FTtoXReturnCode(ftrc); } if(ftrc=TT_Set_Instance_Resolutions(instance->instance, trans->xres, trans->yres)) { TT_Done_Instance(instance->instance); FreeTypeFreeFace(instance->face); xfree(instance); MUMBLE1("Couldn't set resolution: %d\n:", ftrc); return FTtoXReturnCode(ftrc); } if(ftrc=TT_Set_Instance_CharSize(instance->instance, (int)(trans->scale*(1<<6)+0.5))) { TT_Done_Instance(instance->instance); FreeTypeFreeFace(instance->face); xfree(instance); MUMBLE1("Couldn't set charsize: %d\n", ftrc); return FTtoXReturnCode(ftrc); } if(trans->nonIdentity) { int rotated, stretched; stretched = ((trans->matrix.xx>>8)*(trans->matrix.xx>>8) + ((trans->matrix.yx>>8)*(trans->matrix.yx>>8)) != (1<<16)) || ((trans->matrix.xy>>8) * (trans->matrix.xy>>8) + (trans->matrix.yy>>8) * (trans->matrix.yy>>8) != (1<<16)); rotated = trans->matrix.yx!=0 || trans->matrix.xy!=0; TT_Set_Instance_Transform_Flags(instance->instance, rotated, stretched); } if(ftrc=TT_Get_Instance_Metrics(instance->instance, &instance->imetrics)) { TT_Done_Instance(instance->instance); FreeTypeFreeFace(instance->face); xfree(instance); MUMBLE1("Couldn't get instance metrics: %d\n:", ftrc); return FTtoXReturnCode(ftrc); } /* maintain a linked list of instances */ instance->next=instance->face->instances; instance->face->instances=instance; *instancep=instance; return Successful; } static void FreeTypeFreeInstance(TTFInstance *instance) { TTFInstance *otherInstance; instance->refcount--; if(instance->refcount<=0) { int i,j; if(instance->face->instances==instance) instance->face->instances=instance->next; else { for(otherInstance=instance->face->instances; otherInstance; otherInstance=otherInstance->next) if(otherInstance->next==instance) { otherInstance->next=instance->next; break; } } MUMBLE("Freeing instance\n"); TT_Done_Instance(instance->instance); FreeTypeFreeFace(instance->face); if(instance->charcellMetrics) { xfree(instance->charcellMetrics); } if(instance->glyphs) { for(i=0; inglyphs,FONTSEGMENTSIZE); i++) { if(instance->glyphs[i]) { for(j=0; javailable[i][j]==3) xfree(instance->glyphs[i][j].bits); } xfree(instance->glyphs[i]); } } xfree(instance->glyphs); } if(instance->available) { for(i=0; inglyphs,FONTSEGMENTSIZE); i++) { if(instance->available[i]) xfree(instance->available[i]); } xfree(instance->available); } xfree(instance); } } static int FreeTypeInstanceFindGlyph(unsigned idx, TTFInstance *instance, CharInfoPtr **glyphs, int ***available, int *found, int *segmentP, int *offsetP) { int segment, offset; if(idx>instance->nglyphs) { *found=0; return Successful; } if(!*available) { if((*available= (int**)xalloc(sizeof(int*)*iceil(instance->nglyphs, FONTSEGMENTSIZE))) ==0) return AllocError; memset((char*)(*available), 0, sizeof(int*)*iceil(instance->nglyphs, FONTSEGMENTSIZE)); } segment=ifloor(idx, FONTSEGMENTSIZE); offset=idx-segment*FONTSEGMENTSIZE; if(!(*available)[segment]) { if(((*available)[segment]= (int*)xalloc(sizeof(int)*FONTSEGMENTSIZE)) ==0) return AllocError; memset((char*)(*available)[segment],0, sizeof(int)*FONTSEGMENTSIZE); } if(!*glyphs) { if((*glyphs= (CharInfoPtr*)xalloc(sizeof(CharInfoPtr)* iceil(instance->nglyphs, FONTSEGMENTSIZE))) ==0) return AllocError; memset((char*)(*glyphs), 0, sizeof(CharInfoPtr)*iceil(instance->nglyphs, FONTSEGMENTSIZE)); } if(!(*glyphs)[segment]) { if(((*glyphs)[segment]= (CharInfoPtr)xalloc(sizeof(CharInfoRec)*FONTSEGMENTSIZE)) ==0) return AllocError; } *found=1; *segmentP=segment; *offsetP=offset; return Successful; } static int FreeTypeInstanceGetGlyph(unsigned idx, CharInfoPtr *g, TTFInstance *instance) { int found, segment, offset; int xrc, ftrc; int ***available; CharInfoPtr **glyphs; available=&instance->available; glyphs=&instance->glyphs; if((xrc=FreeTypeInstanceFindGlyph(idx, instance, glyphs, available, &found, &segment, &offset)) !=Successful) return xrc; if(!found || (*available)[segment][offset]==1) { *g=0; return Successful; } else if((*available)[segment][offset]==3) { *g=&(*glyphs)[segment][offset]; return Successful; } /* Tough: need to rasterise a new glyph. */ if(ftrc=TT_Load_Glyph(instance->instance, instance->face->glyph, idx, TTLOAD_DEFAULT)) { return FTtoXReturnCode(ftrc); } if(instance->monospaced==2 && (*available)[segment][offset]==0) memcpy((char*)&(*glyphs)[segment][offset].metrics, (char*)instance->charcellMetrics, sizeof(xCharInfo)); xrc=FreeTypeRasteriseGlyph(&(*glyphs)[segment][offset], instance, instance->monospaced==2); if(xrc!=Successful) return xrc; else (*available)[segment][offset]=3; /* Update the width to match the width of the font */ if(instance->monospaced) { if((*available)[segment][offset]>=3) (*glyphs)[segment][offset].metrics.characterWidth=instance->width; } /* return the glyph */ if((*available)[segment][offset]>=3) *g=&(*glyphs)[segment][offset]; else *g=0; return Successful; } static int FreeTypeInstanceGetGlyphMetrics(unsigned idx, xCharInfo **metrics, TTFInstance *instance) { CharInfoPtr g; int xrc; int found, segment, offset; if((xrc=FreeTypeInstanceFindGlyph(idx, instance, &instance->glyphs, &instance->available, &found, &segment, &offset)) !=Successful) return xrc; if(!found) { *metrics=0; return Successful; } if(instance->available[segment][offset]==0) { if(instance->monospaced<2) { /* not a charcell instance */ if((xrc=FreeTypeInstanceGetGlyph(idx, &g, instance))!=Successful) return xrc; } else { memcpy((char*)&instance->glyphs[segment][offset].metrics, (char*)instance->charcellMetrics, sizeof(xCharInfo)); instance->available[segment][offset]=2; } } if(instance->available[segment][offset]>=2) { *metrics=&instance->glyphs[segment][offset].metrics; } else *metrics=0; return Successful; } int FreeTypeRasteriseGlyph(CharInfoPtr tgp, TTFInstance *instance, int hasMetrics) { #define TRANSFORM_X(x_value) \ ((int)floor((((double)(x_value)*(double)instance->transformation.matrix.xx)/\ (TWO_SIXTEENTH*TWO_SIXTH))+0.5)) #define TRANSFORM_Y(y_value) \ ((int)floor((((double)(y_value)*(double)instance->transformation.matrix.yy)/\ (TWO_SIXTEENTH*TWO_SIXTH))+0.5)) #define TRANSFORM_X_RAW(value) \ ((int)floor((double)(value)/instance->transformation.scale/TWO_SIXTH/(instance->imetrics.x_resolution/72.0)*1000.0+0.5)) TTFFace *face; TT_Raster_Map raster; TT_Glyph_Metrics metrics; TT_Outline outline; TT_BBox outline_bbox, *bbox; int ftrc; short xoff, yoff, xoff_pixels, yoff_pixels; int wd, ht, bpr; /* width, height, bytes per row */ int leftSideBearing, rightSideBearing, characterWidth, rawCharacterWidth, ascent, descent; int i=0; face=instance->face; TT_Get_Glyph_Metrics(face->glyph, &metrics); if(instance->transformation.nonIdentity) { TT_Get_Glyph_Outline(face->glyph, &outline); TT_Transform_Outline(&outline, &instance->transformation.matrix); TT_Get_Outline_BBox(&outline, &outline_bbox); bbox=&outline_bbox; } else { bbox=&metrics.bbox; } if(hasMetrics) { xoff=-tgp->metrics.leftSideBearing<<6; yoff=tgp->metrics.descent<<6; wd=tgp->metrics.rightSideBearing - tgp->metrics.leftSideBearing; ht=tgp->metrics.ascent + tgp->metrics.descent; } else { xoff=(63-bbox->xMin) & -64; yoff=(63-bbox->yMin) & -64; wd=(bbox->xMax+63+xoff) >> 6; ht=(bbox->yMax+63+yoff) >> 6; /* The X convention is to consider a character with an empty * bounding box as undefined. This convention is broken. */ if(wd<=0) wd=1; if(ht<=0) ht=1; } /* Compute the number of bytes per row. */ bpr=((wd+(instance->bmfmt.glyph<<3)-1)>>3) & -instance->bmfmt.glyph; if(tgp) { raster.flow = TT_Flow_Down; raster.rows=ht; raster.width=wd; raster.cols=bpr; raster.size = (long)raster.rows*raster.cols; raster.bitmap = (void*)xalloc(ht*raster.cols); if(raster.bitmap==NULL) return AllocError; memset(raster.bitmap,0,(int)raster.size); ftrc=TT_Get_Glyph_Bitmap(face->glyph,&raster,xoff,yoff); if(ftrc) { MUMBLE("Failed to draw bitmap\n"); /* Ignore error, return blank bitmap */ } tgp->bits=raster.bitmap; if(instance->bmfmt.bit==LSBFirst) { BitOrderInvert((unsigned char*)(tgp->bits)+i*ht*bpr, ht*bpr); } if(instance->bmfmt.byte!=instance->bmfmt.bit) switch(instance->bmfmt.scan) { case 1: break; case 2: TwoByteSwap((unsigned char*)(tgp->bits)+i*ht*bpr, ht*bpr); break; case 4: FourByteSwap((unsigned char*)(tgp->bits)+i*ht*bpr, ht*bpr); break; case 8: /* no util function for 64 bits! */ { int j,k; char c, *cp=tgp->bits+i*ht*bpr; for(j=ht*bpr; j>=0; j-=8, cp+=8) for(k=0; k<4; k++) { c=cp[k]; cp[k]=cp[7-k]; cp[7-k]=cp[k]; } } break; default: ; } } if(!hasMetrics) { xoff_pixels = xoff>>6; yoff_pixels = yoff>>6; /* Determine the glyph metrics. */ leftSideBearing = -xoff_pixels; rightSideBearing = wd - xoff_pixels; characterWidth = TRANSFORM_X(metrics.advance); rawCharacterWidth = TRANSFORM_X_RAW(metrics.advance); ascent = ht - yoff_pixels; descent = yoff_pixels; if(tgp) { /* Set the glyph metrics. */ tgp->metrics.attributes = (unsigned short)((short)rawCharacterWidth); tgp->metrics.leftSideBearing = leftSideBearing; tgp->metrics.rightSideBearing = rightSideBearing; tgp->metrics.characterWidth = characterWidth; tgp->metrics.ascent = ascent; tgp->metrics.descent = descent; } } return Successful; #undef TRANSFORM_X #undef TRANSFORM_Y #undef TRANSFORM_X_RAW } static void FreeTypeFreeFont(TTFFont *font) { FreeTypeFreeInstance(font->instance); if(font->ranges) xfree(font->ranges); xfree(font); } /* Free a font. If freeProps is 0, don't free the properties. */ static void FreeTypeFreeXFont(FontPtr pFont, int freeProps) { TTFFont *tf; if(pFont) { if(tf=(TTFFont*)pFont->fontPrivate) { FreeTypeFreeFont(tf); } if(freeProps && pFont->info.nprops>0) { xfree(pFont->info.isStringProp); xfree(pFont->info.props); } DestroyFontRec(pFont); } } /* Unload a font */ static void FreeTypeUnloadXFont(FontPtr pFont) { MUMBLE("Unloading\n"); FreeTypeFreeXFont(pFont, 1); } /* Add the font properties, including the Font name, the XLFD * properties, some strings from the font, and various typographical * data. We only provide data readily available in the tables in the * font for now, altough FIGURE_WIDTH would be a good idea as it is * used by Xaw. */ static int FreeTypeAddProperties(TTFFont *font, FontScalablePtr vals, FontInfoPtr info, char *fontname, int rawAverageWidth) { int i, j, maxprops; char *sp, *ep, val[MAXFONTNAMELEN]; TT_Instance_Metrics imetrics; int upm; /* units per em */ TTFFace *face; TTFInstance *instance; TTFNormalisedTransformation *trans; int xlfdProps=0, hheaProps=0, os2Props=0, postProps=0; instance=font->instance; face=instance->face; trans=&instance->transformation; info->nprops=0; /* in case we abort */ strcpy(val, fontname); if(FontParseXLFDName(val, vals, FONT_XLFD_REPLACE_VALUE)) { xlfdProps=1; } else { MUMBLE("Couldn't parse XLFD\n"); xlfdProps=0; } /* What properties are we going to set? Check what tables there are. */ if(face->properties.header) { upm=face->properties.header->Units_Per_EM; if(face->properties.horizontal) hheaProps=1; if(face->properties.os2) os2Props=1; if(face->properties.postscript) postProps=1; } /* Compute an upper bound on the number of props in order to * allocate the right size of vector. That's because we use * wonderfully flexible data structures. */ maxprops= 1+ /* NAME */ (xlfdProps?14:0)+ /* from XLFD */ (hheaProps?5:0)+ /* from `hhea' table */ 3+ /* from `name' table */ (os2Props?6:0)+ /* from `os/2' table */ (postProps?3:0)+ /* from `post' table */ 2; /* type */ if ((info->props = (FontPropPtr)xalloc(maxprops * sizeof(FontPropRec))) == 0) return AllocError; if ((info->isStringProp = (char*)xalloc(maxprops)) == 0) { xfree(info->props); return AllocError; } memset((char *)info->isStringProp, 0, maxprops); /* Add the FONT property as the very first */ info->props[0].name = MakeAtom("FONT", 4, TRUE); info->props[0].value = MakeAtom(val, strlen(val), TRUE); info->isStringProp[0] = 1; i=1; if(*val && *(sp=val+1)) { for (i = 0, sp=val+1; i < 14; i++) { /* Locate the next field. */ if (i == 13) /* Handle the case of the final field containing a subset * specification. */ for (ep = sp; *ep && *ep != '['; ep++); else for (ep = sp; *ep && *ep != '-'; ep++); /* Create the property name.*/ info->props[i+1].name = MakeAtom(xlfd_props[i], strlen(xlfd_props[i]), TRUE); switch(i) { case 6: /* pixel size */ info->props[i+1].value = (int)(fabs(vals->pixel_matrix[3])+0.5); break; case 7: /* point size */ info->props[i+1].value = (int)(fabs(vals->point_matrix[3])*10.0 + 0.5); break; case 8: /* resolution x */ info->props[i+1].value = vals->x; break; case 9: /* resolution y */ info->props[i+1].value = vals->y; break; case 11: /* average width */ info->props[i+1].value = vals->width; break; default: /* a string */ info->props[i+1].value = MakeAtom(sp, ep - sp, TRUE); info->isStringProp[i+1] = 1; } sp = ++ep; } i++; } /* the following two have already been properly scaled */ if(hheaProps) { info->props[i].name = MakeAtom("RAW_AVERAGE_WIDTH", 17, TRUE); info->props[i].value = rawAverageWidth; i++; info->props[i].name = MakeAtom("FONT_ASCENT", 11, TRUE); info->props[i].value = info->fontAscent; i++; info->props[i].name = MakeAtom("RAW_ASCENT", 15, TRUE); info->props[i].value = ((double)face->properties.horizontal->Ascender/(double)upm*1000.0); i++; info->props[i].name = MakeAtom("FONT_DESCENT", 12, TRUE); info->props[i].value = info->fontDescent; i++; info->props[i].name = MakeAtom("RAW_DESCENT", 16, TRUE); info->props[i].value = -((double)face->properties.horizontal->Descender/(double)upm*1000.0); i++; } if((j=ttf_GetEnglishName(face->face, val, TTF_COPYRIGHT))>0) { info->props[i].name = MakeAtom("COPYRIGHT", 9, TRUE); info->props[i].value = MakeAtom(val, j, TRUE); info->isStringProp[i] = 1; i++; } if((j=ttf_GetEnglishName(face->face, val, TTF_TYPEFACE))>0) { info->props[i].name = MakeAtom("FACE_NAME", 9, TRUE); info->props[i].value = MakeAtom(val, j, TRUE); info->isStringProp[i] = 1; i++; } if((j=ttf_GetEnglishName(face->face, val, TTF_PSNAME))>0) { info->props[i].name = MakeAtom("_ADOBE_POSTSCRIPT_FONTNAME", 26, TRUE); info->props[i].value = MakeAtom(val, j, TRUE); info->isStringProp[i] = 1; i++; } /* These macros handle the case of a diagonal matrix. They convert * FUnits into pixels. */ #define TRANSFORM_FUNITS_X(xval) \ ((int) \ floor((((double)(xval)/(double)upm) * \ ((double)trans->matrix.xx/TWO_SIXTEENTH)*(double)imetrics.x_ppem)+\ 0.5)) #define TRANSFORM_FUNITS_Y(yval) \ ((int) \ floor((((double)(yval)/(double)upm) * \ ((double)trans->matrix.yy/TWO_SIXTEENTH) * (double)imetrics.y_ppem)+\ 0.5)) /* In what follows, we assume the matrix is diagonal. In the rare * case when it is not, the values will be somewhat wrong. */ if(TT_Get_Instance_Metrics(instance->instance, &imetrics)==0) { if(os2Props) { info->props[i].name = MakeAtom("SUBSCRIPT_SIZE",14,TRUE); info->props[i].value = TRANSFORM_FUNITS_Y(face->properties.os2->ySubscriptYSize); i++; info->props[i].name = MakeAtom("SUBSCRIPT_X",11,TRUE); info->props[i].value = TRANSFORM_FUNITS_X(face->properties.os2->ySubscriptXOffset); i++; info->props[i].name = MakeAtom("SUBSCRIPT_Y",11,TRUE); info->props[i].value = TRANSFORM_FUNITS_Y(face->properties.os2->ySubscriptYOffset); i++; info->props[i].name = MakeAtom("SUPERSCRIPT_SIZE",16,TRUE); info->props[i].value = TRANSFORM_FUNITS_Y(face->properties.os2->ySuperscriptYSize); i++; info->props[i].name = MakeAtom("SUPERSCRIPT_X",13,TRUE); info->props[i].value = TRANSFORM_FUNITS_X(face->properties.os2->ySuperscriptXOffset); i++; info->props[i].name = MakeAtom("SUPERSCRIPT_Y",13,TRUE); info->props[i].value = TRANSFORM_FUNITS_Y(face->properties.os2->ySuperscriptYOffset); i++; } if(postProps) { long underlineThickness, underlinePosition; /* tk uses the following two */ info->props[i].name = MakeAtom("UNDERLINE_THICKNESS",19,TRUE); underlineThickness= TRANSFORM_FUNITS_Y(face->properties.postscript->underlineThickness); if(underlineThickness<=0) underlineThickness=1; info->props[i].value = underlineThickness; i++; info->props[i].name = MakeAtom("UNDERLINE_POSITION",18,TRUE); /* PostScript and X use opposite conventions */ underlinePosition= TRANSFORM_FUNITS_Y(-face->properties.postscript->underlinePosition); info->props[i].value = underlinePosition; i++; if(trans->matrix.xx == trans->matrix.yy) { info->props[i].name = MakeAtom("ITALIC_ANGLE",12,TRUE); info->props[i].value = /* Convert from TT_Fixed to * 64th of a degree counterclockwise from 3 o'clock */ 90*64+(face->properties.postscript->italicAngle>>10); i++; } } #undef TRANSFORM_FUNITS_X #undef TRANSFORM_FUNITS_Y } info->props[i].name = MakeAtom("FONT_TYPE", 9, TRUE); info->props[i].value = MakeAtom("TrueType", 8, TRUE); i++; info->props[i].name = MakeAtom("RASTERIZER_NAME", 15, TRUE); info->props[i].value = MakeAtom("FreeType", 8, TRUE); i++; info->nprops=i; return Successful; } static int FreeTypeFontGetGlyph(unsigned code, CharInfoPtr *g, TTFFont *font) { unsigned idx; int i; /* As a special case, we pass 0 even when it is not in the ranges; * this will allow for the default glyph, which should exist in any * TrueType font. */ if(code>0 && font->nranges) { for(i=0; inranges; i++) if((code >= font->ranges[i].min_char_low+ (font->ranges[i].min_char_high<<8)) && (code <= font->ranges[i].max_char_low+(font->ranges[i].max_char_high<<8))) break; if(i==font->nranges) { *g=0; return Successful; } } idx=ttf_remap(code, &font->mapping); /* Only pass the default glyph if there is no first index */ if(idx==0 && (code != 0 || (font->mapping.mapping && (font->mapping.mapping->encoding->first != 0 || font->mapping.mapping->encoding->first_col != 0)))) { *g=0; return Successful; } else { return FreeTypeInstanceGetGlyph(idx, g, font->instance); } } static int FreeTypeFontGetGlyphMetrics(unsigned code, xCharInfo **metrics, TTFFont *font) { unsigned idx; int i; /* As a special case, we pass 0 even when it is not in the ranges; * this will allow for the default glyph, which should exist in any * TrueType font. */ if(code>0 && font->nranges) { for(i=0; inranges; i++) if((code >= font->ranges[i].min_char_low+ (font->ranges[i].min_char_high<<8)) && (code <= font->ranges[i].max_char_low+(font->ranges[i].max_char_high<<8))) break; if(i==font->nranges) { *metrics=0; return Successful; } } idx=ttf_remap(code, &font->mapping); if(idx==0 && (code!=0 || (font->mapping.mapping && (font->mapping.mapping->encoding->first != 0 || font->mapping.mapping->encoding->first_col != 0)))) { *metrics=0; return Successful; } else { return FreeTypeInstanceGetGlyphMetrics(idx, metrics, font->instance); } } static int FreeTypeFontGetDefaultGlyph(CharInfoPtr *g, TTFFont *font) { /* Disable default glyph generation if there is a first index */ if(font->mapping.mapping && (font->mapping.mapping->encoding->first || font->mapping.mapping->encoding->first_col)) { *g=0; return Successful; } /* Using FreeTypeInstanceGetGlyph(0,...) would cause inconsistencies * between metrics and glyphs in the unlikely case that 0 is not * mapped to 0. */ return FreeTypeFontGetGlyph(0, g, font); } static int FreeTypeLoadFont(TTFFont **fontp, char *fileName, FontScalablePtr vals, FontEntryPtr entry, FontBitmapFormat *bmfmt) { int xrc; TTFFont *font; TTFNormalisedTransformation trans; int charcell; if((font=(TTFFont*)xalloc(sizeof(TTFFont)))==NULL) return AllocError; /* Compute the transformation matrix. We use floating-point * arithmetic for simplicity */ trans.xres=vals->x; trans.yres=vals->y; /* This value cannot be 0. */ trans.scale=MAX(hypot(vals->point_matrix[0],vals->point_matrix[2]), hypot(vals->point_matrix[1],vals->point_matrix[3])); trans.nonIdentity=0; /* Try to round stuff. We want approximate zeros to be exact zeros, and if the elements on the diagonal are approximately equal, we want them equal. We do this to avoid breaking hinting. */ if(DIFFER(vals->point_matrix[0], vals->point_matrix[3])) { trans.nonIdentity=1; trans.matrix.xx= (int)((vals->point_matrix[0]*(double)TWO_SIXTEENTH)/trans.scale); trans.matrix.yy= (int)((vals->point_matrix[3]*(double)TWO_SIXTEENTH)/trans.scale); } else { trans.matrix.xx=trans.matrix.yy= ((vals->point_matrix[0]+vals->point_matrix[3])/2* (double)TWO_SIXTEENTH)/trans.scale; } if(DIFFER0(vals->point_matrix[1], trans.scale)) { trans.matrix.yx= (int)((vals->point_matrix[1]*(double)TWO_SIXTEENTH)/trans.scale); trans.nonIdentity=1; } else trans.matrix.yx=0; if(DIFFER0(vals->point_matrix[2], trans.scale)) { trans.matrix.xy= (int)((vals->point_matrix[2]*(double)TWO_SIXTEENTH)/trans.scale); trans.nonIdentity=1; } else trans.matrix.xy=0; /* Check for charcell in XLFD */ charcell=0; if(entry->name.ndashes==14) { char *p; int dashes=0; for(p=entry->name.name; p<=entry->name.name+entry->name.length-2; p++) { if(*p=='-') { dashes++; if(dashes==11) { if(p[1]=='c' && p[2]=='-') charcell=1; break; } } } } if((xrc=FreeTypeOpenInstance(&font->instance, fileName, &trans, charcell, bmfmt)) !=Successful) return xrc; if(entry->name.ndashes==14) { if(ttf_pick_cmap(entry->name.name, entry->name.length, fileName, font->instance->face->face, &font->mapping)) return BadFontFormat; } else { if(ttf_pick_cmap(0, 0, fileName, font->instance->face->face, &font->mapping)) return BadFontFormat; } font->nranges=vals->nranges; font->ranges=0; if(font->nranges) { if((font->ranges=(fsRange*)xalloc(vals->nranges*sizeof(fsRange))) ==NULL) { FreeTypeFreeFont(font); return AllocError; } memcpy((char*)font->ranges, (char*)vals->ranges, vals->nranges*sizeof(fsRange)); } *fontp=font; return Successful; } /* Given a BBox in FUnits, return a transformed BBox in pixels */ static void transformBBox(TTFNormalisedTransformation *transformation, int upm, int x_ppem, int y_ppem, int x1, int y1, int x2, int y2, int *tx1p, int *ty1p, int *tx2p, int *ty2p) { double xx1, yy1, xx2, yy2, tx11, ty11, tx12, ty12, tx21, ty21, tx22, ty22, tx1, ty1, tx2, ty2; /* Convert arguments to EM units */ xx1=((double)x1/(double)upm); yy1=((double)y1/(double)upm); xx2=((double)x2/(double)upm); yy2=((double)y2/(double)upm); /* Apply transformation matrix */ if(!transformation->nonIdentity) { tx1=xx1; ty1=yy1; tx2=xx2; ty2=yy2; } else { /* Not an identity matrix, need to compute images of all corners */ tx11= (transformation->matrix.xx/TWO_SIXTEENTH)*xx1 + (transformation->matrix.xy/TWO_SIXTEENTH)*yy1; ty11= (transformation->matrix.yx/TWO_SIXTEENTH)*xx1 + (transformation->matrix.yy/TWO_SIXTEENTH)*yy1; tx12= (transformation->matrix.xx/TWO_SIXTEENTH)*xx1 + (transformation->matrix.xy/TWO_SIXTEENTH)*yy2; ty12= (transformation->matrix.yx/TWO_SIXTEENTH)*xx1 + (transformation->matrix.yy/TWO_SIXTEENTH)*yy2; tx21= (transformation->matrix.xx/TWO_SIXTEENTH)*xx2 + (transformation->matrix.xy/TWO_SIXTEENTH)*yy1; ty21= (transformation->matrix.yx/TWO_SIXTEENTH)*xx2 + (transformation->matrix.yy/TWO_SIXTEENTH)*yy1; tx22= (transformation->matrix.xx/TWO_SIXTEENTH)*xx2 + (transformation->matrix.xy/TWO_SIXTEENTH)*yy2; ty22= (transformation->matrix.yx/TWO_SIXTEENTH)*xx2 + (transformation->matrix.yy/TWO_SIXTEENTH)*yy2; /* Compute the corners of the new bounding box */ tx1=MIN(MIN(tx11,tx12),MIN(tx21,tx22)); ty1=MIN(MIN(ty11,ty12),MIN(ty21,ty22)); tx2=MAX(MAX(tx11,tx12),MAX(tx21,tx22)); ty2=MAX(MAX(ty11,ty12),MAX(ty21,ty22)); } /* Convert to device space */ *tx1p=(int)floor(tx1*(double)x_ppem); *ty1p=(int)floor(ty1*(double)y_ppem); *tx2p=(int)ceil(tx2*(double)x_ppem); *ty2p=(int)ceil(ty2*(double)y_ppem); /* Ensure the resulting bounding box is not empty */ if(*tx1p==*tx2p) (*tx2p)++; if(*ty1p==*ty2p) (*ty2p)++; } /* Do all the real work for OpenFont or FontInfo */ /* xf->info is only accessed through info, and xf might be null */ static int FreeTypeLoadXFont(char *fileName, FontScalablePtr vals, FontPtr xf, FontInfoPtr info, FontBitmapFormat *bmfmt, FontEntryPtr entry) { #define TRANSFORM_FUNITS_X(xval) \ ((int) \ floor((((double)(xval)/(double)upm) * \ ((double)instance->transformation.matrix.xx/TWO_SIXTEENTH)*(double)imetrics.x_ppem)+0.5)) #define TRANSFORM_FUNITS_X_DOWN(xval) \ ((int) \ floor((((double)(xval)/(double)upm) * \ ((double)instance->transformation.matrix.xx/TWO_SIXTEENTH)*(double)imetrics.x_ppem))) #define TRANSFORM_FUNITS_X_UP(xval) \ ((int) \ ceil((((double)(xval)/(double)upm) * \ ((double)instance->transformation.matrix.xx/TWO_SIXTEENTH)*(double)imetrics.x_ppem))) #define TRANSFORM_FUNITS_Y(yval) \ ((int) \ floor((((double)(yval)/(double)upm) * \ ((double)instance->transformation.matrix.yy/TWO_SIXTEENTH)*(double)imetrics.x_ppem)+0.5)) #define TRANSFORM_FUNITS_Y_DOWN(yval) \ ((int) \ floor((((double)(yval)/(double)upm) * \ ((double)instance->transformation.matrix.yy/TWO_SIXTEENTH) * (double)imetrics.y_ppem))) #define TRANSFORM_FUNITS_Y_UP(yval) \ ((int) \ ceil((((double)(yval)/(double)upm) * \ ((double)instance->transformation.matrix.yy/TWO_SIXTEENTH) * (double)imetrics.y_ppem))) #define TRANSFORM_FUNITS_RAW(value) \ ((long) \ floor(((double)(value)/(double)upm) * 1000.0 + 0.5)) #define TRANSFORM_FUNITS_RAW_DOWN(value) \ ((long) \ floor(((double)(value)/(double)upm) * 1000.0)) #define TRANSFORM_FUNITS_RAW_UP(value) \ ((long) \ ceil(((double)(value)/(double)upm) * 1000.0)) TTFFont *font; TTFInstance *instance; TT_Instance_Metrics imetrics; TTFFace *face; TT_Face_Properties *properties; int xrc, ftrc, i; int charcell, constantWidth; long rawWidth, rawAverageWidth, aw, code, lastCode, firstCode; int upm, minLsb, maxRsb, ascent, descent, width, averageWidth; if((xrc=FreeTypeLoadFont(&font, fileName, vals, entry, bmfmt)) !=Successful) return xrc; instance=font->instance; face=instance->face; properties=&face->properties; if(ftrc=TT_Get_Instance_Metrics(instance->instance, &imetrics)) { FreeTypeFreeFont(font); return FTtoXReturnCode(ftrc); } if(!properties->header || !properties->horizontal) { FreeTypeFreeFont(font); return BadFontFormat; } upm = properties->header->Units_Per_EM; charcell=(instance->monospaced==2); constantWidth= charcell || (properties->postscript && properties->postscript->isFixedPitch); if(constantWidth && instance->monospaced==0) instance->monospaced=1; /* There's no way to get the average width right without rasterising * all of the glyphs. We make a serious attempt at getting it right * for monospaced fonts, and try to provide a reasonable * approximation for others. */ if(constantWidth) aw=properties->horizontal->advance_Width_Max; else if (properties->os2) aw=properties->os2->xAvgCharWidth; else aw=properties->horizontal->advance_Width_Max/2; if(constantWidth) averageWidth = 10*TRANSFORM_FUNITS_X(aw); else averageWidth = TRANSFORM_FUNITS_X(aw*10L); rawAverageWidth = TRANSFORM_FUNITS_RAW(aw*10L); vals->width=averageWidth; if(info) { info->fontAscent = TRANSFORM_FUNITS_Y(properties->horizontal->Ascender); info->fontDescent = -TRANSFORM_FUNITS_Y(properties->horizontal->Descender); firstCode=0; lastCode=0xFFFFL; if(font->nranges) { lastCode=0; /* The ranges information does not have an effect on firstCode, as we pass the default glyph at position 0. */ for(i=0; inranges; i++) { code=font->ranges[i].max_char_low+(font->ranges[i].max_char_high<<8); if(lastCodemapping.mapping || font->mapping.mapping->encoding->row_size == 0) { /* linear indexing */ lastCode=MIN(lastCode, font->mapping.mapping ? font->mapping.mapping->encoding->size-1 : (font->mapping.has_cmap ? 0xFF : 0xFFFF)); if(font->mapping.mapping && font->mapping.mapping->encoding->first) firstCode=font->mapping.mapping->encoding->first; info->firstRow=firstCode/0x100; info->lastRow=lastCode/0x100; info->firstCol= (info->firstRow || info->lastRow) ? 0 : (firstCode & 0xFF); info->lastCol=info->lastRow ? 0xFF : (lastCode & 0xFF); } else { /* matrix indexing */ info->firstRow=font->mapping.mapping->encoding->first; info->lastRow=MIN(font->mapping.mapping->encoding->size-1, lastCode/0x100); info->firstCol=font->mapping.mapping->encoding->first_col; info->lastCol=MIN(font->mapping.mapping->encoding->row_size-1, lastCode<0x100?lastCode:0xFF); } /* firstCode and lastCode are not valid in case of a matrix encoding */ transformBBox(&instance->transformation, upm, instance->imetrics.x_ppem, instance->imetrics.y_ppem, charcell? 0 : properties->header->xMin - properties->horizontal->min_Left_Side_Bearing, properties->header->yMin, charcell ? properties->horizontal->advance_Width_Max : properties->horizontal->xMax_Extent, properties->header->yMax, &minLsb, &descent, &maxRsb, &ascent); descent=-descent; width = TRANSFORM_FUNITS_X(properties->horizontal->advance_Width_Max); rawWidth = TRANSFORM_FUNITS_RAW(properties->horizontal->advance_Width_Max); instance->width=width; info->constantWidth=constantWidth; info->constantMetrics=charcell; info->minbounds.leftSideBearing = minLsb; info->minbounds.rightSideBearing = charcell?maxRsb:minLsb; info->minbounds.characterWidth = constantWidth?width:-width; info->minbounds.ascent = charcell?ascent:-descent; info->minbounds.descent = charcell?descent:-ascent; info->minbounds.attributes = (unsigned short)(short)(constantWidth?rawWidth:-rawWidth); info->maxbounds.leftSideBearing = charcell?minLsb:maxRsb; info->maxbounds.rightSideBearing = maxRsb; info->maxbounds.characterWidth = width; info->maxbounds.ascent = ascent; info->maxbounds.descent = descent; info->maxbounds.attributes = (unsigned short)(short)rawWidth; if(charcell && instance->charcellMetrics==0) { if((instance->charcellMetrics=(xCharInfo*)xalloc(sizeof(xCharInfo))) ==0) { FreeTypeFreeFont(font); return AllocError; } memcpy((char*)instance->charcellMetrics, (char*)&info->maxbounds, sizeof(xCharInfo)); } /* Glyph metrics are accurate */ info->inkMetrics=1; memcpy((char *) &info->ink_maxbounds, (char *) &info->maxbounds, sizeof(xCharInfo)); memcpy((char *) &info->ink_minbounds, (char *) &info->minbounds, sizeof(xCharInfo)); /* XXX - hack */ info->defaultCh=0; } if(xf) xf->fontPrivate=(void*)font; if(info) { if((xrc=FreeTypeAddProperties(font, vals, info, entry->name.name, rawAverageWidth)) !=Successful) { FreeTypeFreeFont(font); return xrc; } } return Successful; #undef TRANSFORM_FUNITS_X #undef TRANSFORM_FUNITS_X_DOWN #undef TRANSFORM_FUNITS_X_UP #undef TRANSFORM_FUNITS_Y #undef TRANSFORM_FUNITS_Y_DOWN #undef TRANSFORM_FUNITS_Y_UP #undef TRANSFORM_FUNITS_RAW #undef TRANSFORM_FUNITS_RAW_DOWN #undef TRANSFORM_FUNITS_RAW_UP } /* Routines used by X11 to get info and glyphs from the font. */ static int FreeTypeGetMetrics(FontPtr pFont, unsigned long count, unsigned char *chars, FontEncoding charEncoding, unsigned long *metricCount, xCharInfo **metrics) { unsigned code; TTFFont *tf; xCharInfo **mp, *m; /* The compiler is supposed to initialise all the fields to 0 */ static xCharInfo noSuchChar; /* MUMBLE1("Get metrics for %ld characters\n", count);*/ tf = (TTFFont*)pFont->fontPrivate; mp = metrics; while (count-- > 0) { switch (charEncoding) { case Linear8Bit: case TwoD8Bit: code = *chars++; break; case Linear16Bit: case TwoD16Bit: code = (*chars++ << 8); code |= *chars++; break; } if(FreeTypeFontGetGlyphMetrics(code, &m, tf)==Successful && m!=0) { *mp++=m; } else *mp++=&noSuchChar; } *metricCount = mp - metrics; return Successful; } static int FreeTypeGetGlyphs(FontPtr pFont, unsigned long count, unsigned char *chars, FontEncoding charEncoding, unsigned long *glyphCount, CharInfoPtr *glyphs) { unsigned code; TTFFont *tf; CharInfoPtr *gp; CharInfoPtr g; tf = (TTFFont*)pFont->fontPrivate; gp = glyphs; while (count-- > 0) { switch (charEncoding) { case Linear8Bit: case TwoD8Bit: code = *chars++; break; case Linear16Bit: case TwoD16Bit: code = *chars++ << 8; code |= *chars++; break; } if(FreeTypeFontGetGlyph(code, &g, tf)==Successful && g!=0) { *gp++ = g; } else if(FreeTypeFontGetDefaultGlyph(&g, tf)==Successful && g!=0) *gp++ = g; } *glyphCount = gp - glyphs; return Successful; } static int FreeTypeSetUpFont(FontPathElementPtr fpe, FontPtr xf, FontInfoPtr info, fsBitmapFormat format, fsBitmapFormatMask fmask, FontBitmapFormat *bmfmt) { int xrc; int image; /* Get the default bitmap format information for this X installation. * Also update it for the client if running in the font server. */ FontDefaultFormat(&bmfmt->bit, &bmfmt->byte, &bmfmt->glyph, &bmfmt->scan); if ((xrc = CheckFSFormat(format, fmask, &bmfmt->bit, &bmfmt->byte, &bmfmt->scan, &bmfmt->glyph, &image)) != Successful) { MUMBLE1("Aborting after checking FS format: %d\n", xrc); return xrc; } if(xf) { xf->refcnt = 0; xf->bit = bmfmt->bit; xf->byte = bmfmt->byte; xf->glyph = bmfmt->glyph; xf->scan = bmfmt->scan; xf->format = format; xf->get_glyphs = FreeTypeGetGlyphs; xf->get_metrics = FreeTypeGetMetrics; xf->unload_font = FreeTypeUnloadXFont; xf->unload_glyphs = 0; xf->fpe = fpe; xf->svrPrivate = 0; xf->fontPrivate = 0; /* we'll set it later */ xf->fpePrivate = 0; } info->defaultCh = 0; info->noOverlap = 0; /* not updated */ info->terminalFont = 0; /* not updated */ info->constantMetrics = 0; /* we'll set it later */ info->constantWidth = 0; /* we'll set it later */ info->inkInside = 1; info->inkMetrics = 1; info->allExist=0; /* not updated */ info->drawDirection = LeftToRight; /* we'll set it later */ info->cachable = 1; /* we don't do licensing */ info->anamorphic = 0; /* can hinting lead to anamorphic scaling? */ info->maxOverlap = 0; /* we'll set it later. */ info->pad = 0; /* ??? */ return Successful; } /* Functions exported by the backend */ static int FreeTypeOpenScalable(FontPathElementPtr fpe, FontPtr *ppFont, int flags, FontEntryPtr entry, char *fileName, FontScalablePtr vals, fsBitmapFormat format, fsBitmapFormatMask fmask, FontPtr non_cachable_font) { int xrc; FontPtr xf; FontBitmapFormat bmfmt; MUMBLE1("Open Scalable %s, XLFD=",fileName); #ifdef DEBUG_TRUETYPE fwrite(entry->name.name, entry->name.length, 1, stdout); #endif MUMBLE("\n"); /* Reject ridiculously small values. Singular matrices are okay. */ if(MAX(hypot(vals->pixel_matrix[0], vals->pixel_matrix[1]), hypot(vals->pixel_matrix[2], vals->pixel_matrix[3])) <1.0) return BadFontName; /* Create an X11 server-side font. */ if (!(xf = CreateFontRec())) return AllocError; if((xrc=FreeTypeSetUpFont(fpe, xf, &xf->info, format, fmask, &bmfmt)) != Successful) { DestroyFontRec(xf); return xrc; } /* Load the font and fill its info structure. */ if ((xrc = FreeTypeLoadXFont(fileName, vals, xf, &xf->info, &bmfmt, entry)) != Successful) { /* Free everything up at this level and return the error code. */ MUMBLE1("Error during load: %d\n",xrc); DestroyFontRec(xf); return xrc; } /* Set the font and return. */ *ppFont = xf; return xrc; } /* Routine to get requested font info. */ static int FreeTypeGetInfoScalable(FontPathElementPtr fpe, FontInfoPtr info, FontEntryPtr entry, FontNamePtr fontName, char *fileName, FontScalablePtr vals) { int xrc; FontBitmapFormat bmfmt; MUMBLE("Get info, XLFD= "); #ifdef DEBUG_TRUETYPE fwrite(entry->name.name, entry->name.length, 1, stdout); #endif MUMBLE("\n"); if(MAX(hypot(vals->pixel_matrix[0], vals->pixel_matrix[1]), hypot(vals->pixel_matrix[2], vals->pixel_matrix[3])) <1.0) return BadFontName; if((xrc=FreeTypeSetUpFont(fpe, 0, info, 0, 0, &bmfmt)) != Successful) { return xrc; } bmfmt.glyph <<= 3; if ((xrc = FreeTypeLoadXFont(fileName, vals, 0, info, &bmfmt, entry)) != Successful) { MUMBLE1("Error during load: %d\n", xrc); return xrc; } return Successful; } /* Renderer registration. */ /* Set the capabilities of this renderer. */ #define CAPABILITIES (CAP_CHARSUBSETTING | CAP_MATRIX) /* Set it up so file names with either upper or lower case can be * loaded. We don't support compressed fonts. */ static FontRendererRec renderers[] = { {".ttf", 4, 0, FreeTypeOpenScalable, 0, FreeTypeGetInfoScalable, 0, CAPABILITIES}, {".TTF", 4, 0, FreeTypeOpenScalable, 0, FreeTypeGetInfoScalable, 0, CAPABILITIES}, {".ttc", 4, 0, FreeTypeOpenScalable, 0, FreeTypeGetInfoScalable, 0, CAPABILITIES}, {".TTC", 4, 0, FreeTypeOpenScalable, 0, FreeTypeGetInfoScalable, 0, CAPABILITIES}, }; static int num_renderers = sizeof(renderers) / sizeof(renderers[0]); void FreeTypeRegisterFontFileFunctions(void) { int i; for (i = 0; i < num_renderers; i++) FontFileRegisterRenderer(&renderers[i]); }