2 * Adobe Font Metric (AFM) file parsing
3 * See http://partners.adobe.com/asn/developer/PDFS/TN/5004.AFM_Spec.pdf
5 * Copyright 1998 Huw D M Davies
11 #include "winnt.h" /* HEAP_ZERO_MEMORY */
14 #include "debugtools.h"
17 DEFAULT_DEBUG_CHANNEL(psdrv);
20 /* ptr to fonts for which we have afm files */
21 FONTFAMILY *PSDRV_AFMFontList = NULL;
24 /***********************************************************
26 * PSDRV_AFMGetCharMetrics
28 * Parses CharMetric section of AFM file.
30 * Actually only collects the widths of numbered chars and puts then in
33 static void PSDRV_AFMGetCharMetrics(AFM *afm, FILE *fp)
35 char line[256], valbuf[256];
36 char *cp, *item, *value, *curpos, *endpos;
40 afm->Metrics = metric = HeapAlloc( PSDRV_Heap, HEAP_ZERO_MEMORY,
41 afm->NumofMetrics * sizeof(AFMMETRICS) );
42 for(i = 0; i < afm->NumofMetrics; i++, metric++) {
44 if(!fgets(line, sizeof(line), fp)) {
45 ERR("Unexpected EOF\n");
48 cp = line + strlen(line);
52 } while(cp > line && isspace(*cp));
59 value = strpbrk(item, " \t");
60 while(isspace(*value))
62 cp = endpos = strchr(value, ';');
65 memcpy(valbuf, value, cp - value + 1);
66 valbuf[cp - value + 1] = '\0';
69 if(!strncmp(item, "C ", 2)) {
70 value = strchr(item, ' ');
71 sscanf(value, " %d", &metric->C);
73 } else if(!strncmp(item, "CH ", 3)) {
74 value = strrchr(item, ' ');
75 sscanf(value, " %x", &metric->C);
78 else if(!strncmp("WX ", item, 3) || !strncmp("W0X ", item, 4)) {
79 sscanf(value, "%f", &metric->WX);
80 if(metric->C >= 0 && metric->C <= 0xff)
81 afm->CharWidths[metric->C] = metric->WX;
84 else if(!strncmp("N ", item, 2)) {
85 strncpy( metric->N, value, sizeof(metric->N) );
88 else if(!strncmp("B ", item, 2)) {
89 sscanf(value, "%f%f%f%f", &metric->B.llx, &metric->B.lly,
90 &metric->B.urx, &metric->B.ury);
92 /* Store height of Aring to use as lfHeight */
93 if(metric->N && !strncmp(metric->N, "Aring", 5))
94 afm->FullAscender = metric->B.ury;
97 /* Ligatures go here... */
103 TRACE("Metrics for '%s' WX = %f B = %f,%f - %f,%f\n",
104 metric->N, metric->WX, metric->B.llx, metric->B.lly,
105 metric->B.urx, metric->B.ury);
112 /***********************************************************
116 * Fills out an AFM structure and associated substructures (see psdrv.h)
117 * for a given AFM file. All memory is allocated from the process heap.
118 * Returns a ptr to the AFM structure or NULL on error.
120 * This is not complete (we don't handle kerning yet) and not efficient
122 static AFM *PSDRV_AFMParse(char const *file)
130 TRACE("parsing '%s'\n", file);
132 if((fp = fopen(file, "r")) == NULL) {
133 MESSAGE("Can't open AFM file '%s'. Please check wine.conf .\n", file);
137 afm = HeapAlloc(PSDRV_Heap, HEAP_ZERO_MEMORY, sizeof(AFM));
143 while(fgets(buf, sizeof(buf), fp)) {
144 cp = buf + strlen(buf);
148 } while(cp > buf && isspace(*cp));
150 value = strchr(buf, ' ');
152 while(isspace(*value))
155 if(!strncmp("FontName", buf, 8)) {
156 afm->FontName = HEAP_strdupA(PSDRV_Heap, 0, value);
160 if(!strncmp("FullName", buf, 8)) {
161 afm->FullName = HEAP_strdupA(PSDRV_Heap, 0, value);
165 if(!strncmp("FamilyName", buf, 10)) {
166 afm->FamilyName = HEAP_strdupA(PSDRV_Heap, 0, value);
170 if(!strncmp("Weight", buf, 6)) {
171 if(!strncmp("Roman", value, 5) || !strncmp("Medium", value, 6)
172 || !strncmp("Book", value, 4) || !strncmp("Regular", value, 7)
173 || !strncmp("Normal", value, 6))
174 afm->Weight = FW_NORMAL;
175 else if(!strncmp("Demi", value, 4))
176 afm->Weight = FW_DEMIBOLD;
177 else if(!strncmp("Bold", value, 4))
178 afm->Weight = FW_BOLD;
179 else if(!strncmp("Light", value, 5))
180 afm->Weight = FW_LIGHT;
181 else if(!strncmp("Black", value, 5))
182 afm->Weight = FW_BLACK;
184 FIXME("Unkown AFM Weight '%s'\n", value);
185 afm->Weight = FW_NORMAL;
190 if(!strncmp("ItalicAngle", buf, 11)) {
191 sscanf(value, "%f", &(afm->ItalicAngle));
195 if(!strncmp("IsFixedPitch", buf, 12)) {
196 if(!strncasecmp("false", value, 5))
197 afm->IsFixedPitch = FALSE;
199 afm->IsFixedPitch = TRUE;
203 if(!strncmp("FontBBox", buf, 8)) {
204 sscanf(value, "%f %f %f %f", &(afm->FontBBox.llx),
205 &(afm->FontBBox.lly), &(afm->FontBBox.urx),
206 &(afm->FontBBox.ury) );
210 if(!strncmp("UnderlinePosition", buf, 17)) {
211 sscanf(value, "%f", &(afm->UnderlinePosition) );
215 if(!strncmp("UnderlineThickness", buf, 18)) {
216 sscanf(value, "%f", &(afm->UnderlineThickness) );
220 if(!strncmp("CapHeight", buf, 9)) {
221 sscanf(value, "%f", &(afm->CapHeight) );
225 if(!strncmp("XHeight", buf, 7)) {
226 sscanf(value, "%f", &(afm->XHeight) );
230 if(!strncmp("Ascender", buf, 8)) {
231 sscanf(value, "%f", &(afm->Ascender) );
235 if(!strncmp("Descender", buf, 9)) {
236 sscanf(value, "%f", &(afm->Descender) );
240 if(!strncmp("StartCharMetrics", buf, 16)) {
241 sscanf(value, "%d", &(afm->NumofMetrics) );
242 PSDRV_AFMGetCharMetrics(afm, fp);
246 if(!strncmp("EncodingScheme", buf, 14)) {
247 afm->EncodingScheme = HEAP_strdupA(PSDRV_Heap, 0, value);
254 if(afm->FontName == NULL)
255 WARN("%s contains no FontName.\n", file);
256 if(afm->FullName == NULL)
257 afm->FullName = HEAP_strdupA(PSDRV_Heap, 0, afm->FontName);
258 if(afm->FamilyName == NULL)
259 afm->FamilyName = HEAP_strdupA(PSDRV_Heap, 0, afm->FontName);
260 if(afm->Ascender == 0.0)
261 afm->Ascender = afm->FontBBox.ury;
262 if(afm->Descender == 0.0)
263 afm->Descender = afm->FontBBox.lly;
264 if(afm->FullAscender == 0.0)
265 afm->FullAscender = afm->Ascender;
267 afm->Weight = FW_NORMAL;
272 /***********************************************************
276 * Frees the family and afmlistentry structures in list head
278 void PSDRV_FreeAFMList( FONTFAMILY *head )
280 AFMLISTENTRY *afmle, *nexta;
281 FONTFAMILY *family, *nextf;
283 for(nextf = family = head; nextf; family = nextf) {
284 for(nexta = afmle = family->afmlist; nexta; afmle = nexta) {
286 HeapFree( PSDRV_Heap, 0, afmle );
288 nextf = family->next;
289 HeapFree( PSDRV_Heap, 0, family );
295 /***********************************************************
297 * PSDRV_FindAFMinList
298 * Returns ptr to an AFM if name (which is a PS font name) exists in list
301 AFM *PSDRV_FindAFMinList(FONTFAMILY *head, char *name)
306 for(family = head; family; family = family->next) {
307 for(afmle = family->afmlist; afmle; afmle = afmle->next) {
308 if(!strcmp(afmle->afm->FontName, name))
315 /***********************************************************
319 * Adds an afm to the list whose head is pointed to by head. Creates new
320 * family node if necessary and always creates a new AFMLISTENTRY.
322 void PSDRV_AddAFMtoList(FONTFAMILY **head, AFM *afm)
324 FONTFAMILY *family = *head;
325 FONTFAMILY **insert = head;
326 AFMLISTENTRY *tmpafmle, *newafmle;
328 newafmle = HeapAlloc(PSDRV_Heap, HEAP_ZERO_MEMORY,
333 if(!strcmp(family->FamilyName, afm->FamilyName))
335 insert = &(family->next);
336 family = family->next;
340 family = HeapAlloc(PSDRV_Heap, HEAP_ZERO_MEMORY,
343 family->FamilyName = HEAP_strdupA(PSDRV_Heap, 0,
345 family->afmlist = newafmle;
349 tmpafmle = family->afmlist;
350 while(tmpafmle->next)
351 tmpafmle = tmpafmle->next;
353 tmpafmle->next = newafmle;
358 /**********************************************************
360 * PSDRV_ReencodeCharWidths
362 * Re map the CharWidths field of the afm to correspond to an ANSI encoding
365 static void PSDRV_ReencodeCharWidths(AFM *afm)
370 for(i = 0; i < 256; i++) {
373 if(PSDRV_ANSIVector[i] == NULL) {
374 afm->CharWidths[i] = 0.0;
377 for (j = 0, metric = afm->Metrics; j < afm->NumofMetrics; j++, metric++) {
378 if(!strcmp(metric->N, PSDRV_ANSIVector[i])) {
379 afm->CharWidths[i] = metric->WX;
383 if(j == afm->NumofMetrics) {
384 WARN("Couldn't find glyph '%s' in font '%s'\n",
385 PSDRV_ANSIVector[i], afm->FontName);
386 afm->CharWidths[i] = 0.0;
393 /***********************************************************
398 static void PSDRV_DumpFontList(void)
403 for(family = PSDRV_AFMFontList; family; family = family->next) {
404 TRACE("Family '%s'\n", family->FamilyName);
405 for(afmle = family->afmlist; afmle; afmle = afmle->next) {
406 TRACE("\tFontName '%s'\n", afmle->afm->FontName);
413 /***********************************************************
415 * PSDRV_GetFontMetrics
417 * Only exported function in this file. Parses all afm files listed in
418 * [afmfiles] of wine.conf .
420 BOOL PSDRV_GetFontMetrics(void)
426 while (PROFILE_EnumWineIniString( "afmfiles", idx++, key, sizeof(key), value, sizeof(value)))
428 AFM *afm = PSDRV_AFMParse(value);
431 if(afm->EncodingScheme &&
432 !strcmp(afm->EncodingScheme, "AdobeStandardEncoding")) {
433 PSDRV_ReencodeCharWidths(afm);
435 PSDRV_AddAFMtoList(&PSDRV_AFMFontList, afm);
438 PSDRV_DumpFontList();