Bringing apdf from vendor into main branch.
[AROS-Contrib.git] / apdf / freetype2 / sfnt / ttpost.c
blob686a987641571521467c81e86d2e5e8d0fc2e81c
1 /***************************************************************************/
2 /* */
3 /* ttpost.c */
4 /* */
5 /* Postcript name table processing for TrueType and OpenType fonts */
6 /* (body). */
7 /* */
8 /* Copyright 1996-2001, 2002 by */
9 /* David Turner, Robert Wilhelm, and Werner Lemberg. */
10 /* */
11 /* This file is part of the FreeType project, and may only be used, */
12 /* modified, and distributed under the terms of the FreeType project */
13 /* license, LICENSE.TXT. By continuing to use, modify, or distribute */
14 /* this file you indicate that you have read the license and */
15 /* understand and accept it fully. */
16 /* */
17 /***************************************************************************/
19 /*************************************************************************/
20 /* */
21 /* The post table is not completely loaded by the core engine. This */
22 /* file loads the missing PS glyph names and implements an API to access */
23 /* them. */
24 /* */
25 /*************************************************************************/
28 #include <ft2build.h>
29 #include FT_INTERNAL_STREAM_H
30 #include FT_TRUETYPE_TAGS_H
31 #include "ttpost.h"
32 #include "ttload.h"
34 #include "sferrors.h"
37 /*************************************************************************/
38 /* */
39 /* The macro FT_COMPONENT is used in trace mode. It is an implicit */
40 /* parameter of the FT_TRACE() and FT_ERROR() macros, used to print/log */
41 /* messages during execution. */
42 /* */
43 #undef FT_COMPONENT
44 #define FT_COMPONENT trace_ttpost
47 /* If this configuration macro is defined, we rely on the `PSNames' */
48 /* module to grab the glyph names. */
50 #ifdef FT_CONFIG_OPTION_POSTSCRIPT_NAMES
53 #include FT_INTERNAL_POSTSCRIPT_NAMES_H
55 #define MAC_NAME( x ) ( (FT_String*)psnames->macintosh_name( x ) )
58 #else /* FT_CONFIG_OPTION_POSTSCRIPT_NAMES */
61 /* Otherwise, we ignore the `PSNames' module, and provide our own */
62 /* table of Mac names. Thus, it is possible to build a version of */
63 /* FreeType without the Type 1 driver & PSNames module. */
65 #define MAC_NAME( x ) TT_Post_Default_Names[x]
67 /* the 258 default Mac PS glyph names */
69 FT_String* TT_Post_Default_Names[258] =
71 /* 0 */
72 ".notdef", ".null", "CR", "space", "exclam",
73 "quotedbl", "numbersign", "dollar", "percent", "ampersand",
74 /* 10 */
75 "quotesingle", "parenleft", "parenright", "asterisk", "plus",
76 "comma", "hyphen", "period", "slash", "zero",
77 /* 20 */
78 "one", "two", "three", "four", "five",
79 "six", "seven", "eight", "nine", "colon",
80 /* 30 */
81 "semicolon", "less", "equal", "greater", "question",
82 "at", "A", "B", "C", "D",
83 /* 40 */
84 "E", "F", "G", "H", "I",
85 "J", "K", "L", "M", "N",
86 /* 50 */
87 "O", "P", "Q", "R", "S",
88 "T", "U", "V", "W", "X",
89 /* 60 */
90 "Y", "Z", "bracketleft", "backslash", "bracketright",
91 "asciicircum", "underscore", "grave", "a", "b",
92 /* 70 */
93 "c", "d", "e", "f", "g",
94 "h", "i", "j", "k", "l",
95 /* 80 */
96 "m", "n", "o", "p", "q",
97 "r", "s", "t", "u", "v",
98 /* 90 */
99 "w", "x", "y", "z", "braceleft",
100 "bar", "braceright", "asciitilde", "Adieresis", "Aring",
101 /* 100 */
102 "Ccedilla", "Eacute", "Ntilde", "Odieresis", "Udieresis",
103 "aacute", "agrave", "acircumflex", "adieresis", "atilde",
104 /* 110 */
105 "aring", "ccedilla", "eacute", "egrave", "ecircumflex",
106 "edieresis", "iacute", "igrave", "icircumflex", "idieresis",
107 /* 120 */
108 "ntilde", "oacute", "ograve", "ocircumflex", "odieresis",
109 "otilde", "uacute", "ugrave", "ucircumflex", "udieresis",
110 /* 130 */
111 "dagger", "degree", "cent", "sterling", "section",
112 "bullet", "paragraph", "germandbls", "registered", "copyright",
113 /* 140 */
114 "trademark", "acute", "dieresis", "notequal", "AE",
115 "Oslash", "infinity", "plusminus", "lessequal", "greaterequal",
116 /* 150 */
117 "yen", "mu", "partialdiff", "summation", "product",
118 "pi", "integral", "ordfeminine", "ordmasculine", "Omega",
119 /* 160 */
120 "ae", "oslash", "questiondown", "exclamdown", "logicalnot",
121 "radical", "florin", "approxequal", "Delta", "guillemotleft",
122 /* 170 */
123 "guillemotright", "ellipsis", "nbspace", "Agrave", "Atilde",
124 "Otilde", "OE", "oe", "endash", "emdash",
125 /* 180 */
126 "quotedblleft", "quotedblright", "quoteleft", "quoteright", "divide",
127 "lozenge", "ydieresis", "Ydieresis", "fraction", "currency",
128 /* 190 */
129 "guilsinglleft", "guilsinglright", "fi", "fl", "daggerdbl",
130 "periodcentered", "quotesinglbase", "quotedblbase", "perthousand", "Acircumflex",
131 /* 200 */
132 "Ecircumflex", "Aacute", "Edieresis", "Egrave", "Iacute",
133 "Icircumflex", "Idieresis", "Igrave", "Oacute", "Ocircumflex",
134 /* 210 */
135 "apple", "Ograve", "Uacute", "Ucircumflex", "Ugrave",
136 "dotlessi", "circumflex", "tilde", "macron", "breve",
137 /* 220 */
138 "dotaccent", "ring", "cedilla", "hungarumlaut", "ogonek",
139 "caron", "Lslash", "lslash", "Scaron", "scaron",
140 /* 230 */
141 "Zcaron", "zcaron", "brokenbar", "Eth", "eth",
142 "Yacute", "yacute", "Thorn", "thorn", "minus",
143 /* 240 */
144 "multiply", "onesuperior", "twosuperior", "threesuperior", "onehalf",
145 "onequarter", "threequarters", "franc", "Gbreve", "gbreve",
146 /* 250 */
147 "Idot", "Scedilla", "scedilla", "Cacute", "cacute",
148 "Ccaron", "ccaron", "dmacron",
152 #endif /* FT_CONFIG_OPTION_POSTSCRIPT_NAMES */
155 static FT_Error
156 Load_Format_20( TT_Face face,
157 FT_Stream stream )
159 FT_Memory memory = stream->memory;
160 FT_Error error;
162 FT_Int num_glyphs;
163 FT_UShort num_names;
165 FT_UShort* glyph_indices = 0;
166 FT_Char** name_strings = 0;
169 if ( FT_READ_USHORT( num_glyphs ) )
170 goto Exit;
172 /* UNDOCUMENTED! The number of glyphs in this table can be smaller */
173 /* than the value in the maxp table (cf. cyberbit.ttf). */
175 /* There already exist fonts which have more than 32768 glyph names */
176 /* in this table, so the test for this threshold has been dropped. */
178 if ( num_glyphs > face->root.num_glyphs )
180 error = SFNT_Err_Invalid_File_Format;
181 goto Exit;
184 /* load the indices */
186 FT_Int n;
189 if ( FT_NEW_ARRAY ( glyph_indices, num_glyphs ) ||
190 FT_FRAME_ENTER( num_glyphs * 2L ) )
191 goto Fail;
193 for ( n = 0; n < num_glyphs; n++ )
194 glyph_indices[n] = FT_GET_USHORT();
196 FT_FRAME_EXIT();
199 /* compute number of names stored in table */
201 FT_Int n;
204 num_names = 0;
206 for ( n = 0; n < num_glyphs; n++ )
208 FT_Int idx;
211 idx = glyph_indices[n];
212 if ( idx >= 258 )
214 idx -= 257;
215 if ( idx > num_names )
216 num_names = (FT_UShort)idx;
221 /* now load the name strings */
223 FT_UShort n;
226 if ( FT_NEW_ARRAY( name_strings, num_names ) )
227 goto Fail;
229 for ( n = 0; n < num_names; n++ )
231 FT_UInt len;
234 if ( FT_READ_BYTE ( len ) ||
235 FT_NEW_ARRAY( name_strings[n], len + 1 ) ||
236 FT_STREAM_READ ( name_strings[n], len ) )
237 goto Fail1;
239 name_strings[n][len] = '\0';
243 /* all right, set table fields and exit successfuly */
245 TT_Post_20 table = &face->postscript_names.names.format_20;
248 table->num_glyphs = (FT_UShort)num_glyphs;
249 table->num_names = (FT_UShort)num_names;
250 table->glyph_indices = glyph_indices;
251 table->glyph_names = name_strings;
253 return SFNT_Err_Ok;
255 Fail1:
257 FT_UShort n;
260 for ( n = 0; n < num_names; n++ )
261 FT_FREE( name_strings[n] );
264 Fail:
265 FT_FREE( name_strings );
266 FT_FREE( glyph_indices );
268 Exit:
269 return error;
273 static FT_Error
274 Load_Format_25( TT_Face face,
275 FT_Stream stream )
277 FT_Memory memory = stream->memory;
278 FT_Error error;
280 FT_Int num_glyphs;
281 FT_Char* offset_table = 0;
284 /* UNDOCUMENTED! This value appears only in the Apple TT specs. */
285 if ( FT_READ_USHORT( num_glyphs ) )
286 goto Exit;
288 /* check the number of glyphs */
289 if ( num_glyphs > face->root.num_glyphs || num_glyphs > 258 )
291 error = SFNT_Err_Invalid_File_Format;
292 goto Exit;
295 if ( FT_ALLOC( offset_table, num_glyphs ) ||
296 FT_STREAM_READ( offset_table, num_glyphs ) )
297 goto Fail;
299 /* now check the offset table */
301 FT_Int n;
304 for ( n = 0; n < num_glyphs; n++ )
306 FT_Long idx = (FT_Long)n + offset_table[n];
309 if ( idx < 0 || idx > num_glyphs )
311 error = SFNT_Err_Invalid_File_Format;
312 goto Fail;
317 /* OK, set table fields and exit successfuly */
319 TT_Post_25 table = &face->postscript_names.names.format_25;
322 table->num_glyphs = (FT_UShort)num_glyphs;
323 table->offsets = offset_table;
326 return SFNT_Err_Ok;
328 Fail:
329 FT_FREE( offset_table );
331 Exit:
332 return error;
336 static FT_Error
337 Load_Post_Names( TT_Face face )
339 FT_Stream stream;
340 FT_Error error;
341 FT_Fixed format;
344 /* get a stream for the face's resource */
345 stream = face->root.stream;
347 /* seek to the beginning of the PS names table */
348 error = face->goto_table( face, TTAG_post, stream, 0 );
349 if ( error )
350 goto Exit;
352 format = face->postscript.FormatType;
354 /* go to beginning of subtable */
355 if ( FT_STREAM_SKIP( 32 ) )
356 goto Exit;
358 /* now read postscript table */
359 switch ( format )
361 case 0x00020000L:
362 error = Load_Format_20( face, stream );
363 break;
365 case 0x00028000L:
366 error = Load_Format_25( face, stream );
367 break;
369 default:
370 error = SFNT_Err_Invalid_File_Format;
373 face->postscript_names.loaded = 1;
375 Exit:
376 return error;
380 FT_LOCAL_DEF( void )
381 TT_Free_Post_Names( TT_Face face )
383 FT_Memory memory = face->root.memory;
384 TT_Post_Names names = &face->postscript_names;
387 if ( names->loaded )
389 switch ( face->postscript.FormatType )
391 case 0x00020000L:
393 TT_Post_20 table = &names->names.format_20;
394 FT_UShort n;
397 FT_FREE( table->glyph_indices );
398 table->num_glyphs = 0;
400 for ( n = 0; n < table->num_names; n++ )
401 FT_FREE( table->glyph_names[n] );
403 FT_FREE( table->glyph_names );
404 table->num_names = 0;
406 break;
408 case 0x00028000L:
410 TT_Post_25 table = &names->names.format_25;
413 FT_FREE( table->offsets );
414 table->num_glyphs = 0;
416 break;
419 names->loaded = 0;
423 /*************************************************************************/
424 /* */
425 /* <Function> */
426 /* TT_Get_PS_Name */
427 /* */
428 /* <Description> */
429 /* Gets the PostScript glyph name of a glyph. */
430 /* */
431 /* <Input> */
432 /* face :: A handle to the parent face. */
433 /* */
434 /* idx :: The glyph index. */
435 /* */
436 /* PSname :: The address of a string pointer. Will be NULL in case */
437 /* of error, otherwise it is a pointer to the glyph name. */
438 /* */
439 /* You must not modify the returned string! */
440 /* */
441 /* <Output> */
442 /* FreeType error code. 0 means success. */
443 /* */
444 FT_LOCAL_DEF( FT_Error )
445 TT_Get_PS_Name( TT_Face face,
446 FT_UInt idx,
447 FT_String** PSname )
449 FT_Error error;
450 TT_Post_Names names;
452 #ifdef FT_CONFIG_OPTION_POSTSCRIPT_NAMES
453 PSNames_Service psnames;
454 #endif
457 if ( !face )
458 return SFNT_Err_Invalid_Face_Handle;
460 if ( idx >= (FT_UInt)face->root.num_glyphs )
461 return SFNT_Err_Invalid_Glyph_Index;
463 #ifdef FT_CONFIG_OPTION_POSTSCRIPT_NAMES
464 psnames = (PSNames_Service)face->psnames;
465 if ( !psnames )
466 return SFNT_Err_Unimplemented_Feature;
467 #endif
469 names = &face->postscript_names;
471 /* `.notdef' by default */
472 *PSname = MAC_NAME( 0 );
474 switch ( face->postscript.FormatType )
476 case 0x00010000L:
477 if ( idx < 258 ) /* paranoid checking */
478 *PSname = MAC_NAME( idx );
479 break;
481 case 0x00020000L:
483 TT_Post_20 table = &names->names.format_20;
486 if ( !names->loaded )
488 error = Load_Post_Names( face );
489 if ( error )
490 break;
493 if ( idx < (FT_UInt)table->num_glyphs )
495 FT_UShort name_index = table->glyph_indices[idx];
498 if ( name_index < 258 )
499 *PSname = MAC_NAME( name_index );
500 else
501 *PSname = (FT_String*)table->glyph_names[name_index - 258];
504 break;
506 case 0x00028000L:
508 TT_Post_25 table = &names->names.format_25;
511 if ( !names->loaded )
513 error = Load_Post_Names( face );
514 if ( error )
515 break;
518 if ( idx < (FT_UInt)table->num_glyphs ) /* paranoid checking */
520 idx += table->offsets[idx];
521 *PSname = MAC_NAME( idx );
524 break;
526 case 0x00030000L:
527 break; /* nothing to do */
530 return SFNT_Err_Ok;
534 /* END */