David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1 | /***************************************************************************/ |
| 2 | /* */ |
| 3 | /* ttload.c */ |
| 4 | /* */ |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 5 | /* Load the basic TrueType tables, i.e., tables that can be either in */ |
| 6 | /* TTF or OTF fonts (body). */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 7 | /* */ |
Werner Lemberg | 415235d | 2001-06-28 17:49:10 +0000 | [diff] [blame] | 8 | /* Copyright 1996-2001 by */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 9 | /* David Turner, Robert Wilhelm, and Werner Lemberg. */ |
| 10 | /* */ |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 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 */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 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 | /* */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 17 | /***************************************************************************/ |
| 18 | |
Werner Lemberg | cc069be | 2000-12-08 16:17:16 +0000 | [diff] [blame] | 19 | |
David Turner | 19ed8af | 2000-12-08 02:42:29 +0000 | [diff] [blame] | 20 | #include <ft2build.h> |
| 21 | #include FT_INTERNAL_DEBUG_H |
David Turner | 19ed8af | 2000-12-08 02:42:29 +0000 | [diff] [blame] | 22 | #include FT_INTERNAL_STREAM_H |
| 23 | #include FT_TRUETYPE_TAGS_H |
David Turner | 8d3a401 | 2001-03-20 11:14:24 +0000 | [diff] [blame] | 24 | #include "ttload.h" |
| 25 | #include "ttcmap.h" |
David Turner | a90663f | 2000-07-08 00:41:13 +0000 | [diff] [blame] | 26 | |
Werner Lemberg | 1f7f0e8 | 2001-06-06 17:30:41 +0000 | [diff] [blame] | 27 | #include "sferrors.h" |
| 28 | |
David Turner | 9a554eb | 2001-06-27 12:40:46 +0000 | [diff] [blame] | 29 | #include <stdlib.h> /* for qsort */ |
Werner Lemberg | cc069be | 2000-12-08 16:17:16 +0000 | [diff] [blame] | 30 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 31 | /*************************************************************************/ |
| 32 | /* */ |
| 33 | /* The macro FT_COMPONENT is used in trace mode. It is an implicit */ |
| 34 | /* parameter of the FT_TRACE() and FT_ERROR() macros, used to print/log */ |
| 35 | /* messages during execution. */ |
| 36 | /* */ |
| 37 | #undef FT_COMPONENT |
| 38 | #define FT_COMPONENT trace_ttload |
David Turner | a56489e | 2000-02-13 13:41:56 +0000 | [diff] [blame] | 39 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 40 | |
| 41 | /*************************************************************************/ |
| 42 | /* */ |
| 43 | /* <Function> */ |
| 44 | /* TT_LookUp_Table */ |
| 45 | /* */ |
| 46 | /* <Description> */ |
| 47 | /* Looks for a TrueType table by name. */ |
| 48 | /* */ |
| 49 | /* <Input> */ |
| 50 | /* face :: A face object handle. */ |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 51 | /* tag :: The searched tag. */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 52 | /* */ |
| 53 | /* <Return> */ |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 54 | /* A pointer to the table directory entry. 0 if not found. */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 55 | /* */ |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 56 | FT_LOCAL_DEF( TT_Table ) |
Werner Lemberg | 52005c3 | 2001-06-27 23:25:46 +0000 | [diff] [blame] | 57 | TT_LookUp_Table( TT_Face face, |
| 58 | FT_ULong tag ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 59 | { |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 60 | TT_Table entry; |
| 61 | TT_Table limit; |
David Turner | e49ab25 | 2000-05-16 23:44:38 +0000 | [diff] [blame] | 62 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 63 | |
Tom Kacvinsky | 200a1a4 | 2000-11-13 07:31:21 +0000 | [diff] [blame] | 64 | FT_TRACE3(( "TT_LookUp_Table: %08p, `%c%c%c%c' -- ", |
Werner Lemberg | 34e5270 | 2000-11-13 16:58:01 +0000 | [diff] [blame] | 65 | face, |
| 66 | (FT_Char)( tag >> 24 ), |
| 67 | (FT_Char)( tag >> 16 ), |
| 68 | (FT_Char)( tag >> 8 ), |
| 69 | (FT_Char)( tag ) )); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 70 | |
| 71 | entry = face->dir_tables; |
| 72 | limit = entry + face->num_tables; |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 73 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 74 | for ( ; entry < limit; entry++ ) |
| 75 | { |
| 76 | if ( entry->Tag == tag ) |
Tom Kacvinsky | 200a1a4 | 2000-11-13 07:31:21 +0000 | [diff] [blame] | 77 | { |
Werner Lemberg | 34e5270 | 2000-11-13 16:58:01 +0000 | [diff] [blame] | 78 | FT_TRACE3(( "found table.\n" )); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 79 | return entry; |
Tom Kacvinsky | 200a1a4 | 2000-11-13 07:31:21 +0000 | [diff] [blame] | 80 | } |
David Turner | e49ab25 | 2000-05-16 23:44:38 +0000 | [diff] [blame] | 81 | } |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 82 | |
Tom Kacvinsky | 200a1a4 | 2000-11-13 07:31:21 +0000 | [diff] [blame] | 83 | FT_TRACE3(( "could not find table!\n" )); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 84 | return 0; |
| 85 | } |
| 86 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 87 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 88 | /*************************************************************************/ |
| 89 | /* */ |
| 90 | /* <Function> */ |
| 91 | /* TT_Goto_Table */ |
| 92 | /* */ |
| 93 | /* <Description> */ |
| 94 | /* Looks for a TrueType table by name, then seek a stream to it. */ |
| 95 | /* */ |
| 96 | /* <Input> */ |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 97 | /* face :: A face object handle. */ |
| 98 | /* tag :: The searched tag. */ |
| 99 | /* stream :: The stream to seek when the table is found. */ |
| 100 | /* */ |
| 101 | /* <Output> */ |
| 102 | /* length :: The length of the table if found, undefined otherwise. */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 103 | /* */ |
| 104 | /* <Return> */ |
Werner Lemberg | a929ba9 | 2000-06-25 06:47:11 +0000 | [diff] [blame] | 105 | /* FreeType error code. 0 means success. */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 106 | /* */ |
David Turner | bc82f1b | 2002-03-01 02:26:22 +0000 | [diff] [blame] | 107 | FT_LOCAL_DEF( FT_Error ) |
Werner Lemberg | 52005c3 | 2001-06-27 23:25:46 +0000 | [diff] [blame] | 108 | TT_Goto_Table( TT_Face face, |
| 109 | FT_ULong tag, |
| 110 | FT_Stream stream, |
| 111 | FT_ULong* length ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 112 | { |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 113 | TT_Table table; |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 114 | FT_Error error; |
David Turner | e49ab25 | 2000-05-16 23:44:38 +0000 | [diff] [blame] | 115 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 116 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 117 | table = TT_LookUp_Table( face, tag ); |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 118 | if ( table ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 119 | { |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 120 | if ( length ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 121 | *length = table->Length; |
David Turner | e49ab25 | 2000-05-16 23:44:38 +0000 | [diff] [blame] | 122 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 123 | if ( FT_STREAM_SEEK( table->Offset ) ) |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 124 | goto Exit; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 125 | } |
| 126 | else |
Werner Lemberg | 1f7f0e8 | 2001-06-06 17:30:41 +0000 | [diff] [blame] | 127 | error = SFNT_Err_Table_Missing; |
David Turner | e49ab25 | 2000-05-16 23:44:38 +0000 | [diff] [blame] | 128 | |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 129 | Exit: |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 130 | return error; |
| 131 | } |
| 132 | |
| 133 | |
| 134 | /*************************************************************************/ |
| 135 | /* */ |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 136 | /* <Function> */ |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 137 | /* TT_Load_SFNT_HeaderRec */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 138 | /* */ |
| 139 | /* <Description> */ |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 140 | /* Loads the header of a SFNT font file. Supports collections. */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 141 | /* */ |
| 142 | /* <Input> */ |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 143 | /* face :: A handle to the target face object. */ |
| 144 | /* stream :: The input stream. */ |
| 145 | /* face_index :: If the font is a collection, the number of the font */ |
| 146 | /* in the collection, ignored otherwise. */ |
David Turner | b1677a8 | 2000-05-29 20:37:41 +0000 | [diff] [blame] | 147 | /* */ |
David Turner | d42c68e | 2000-01-27 13:56:02 +0000 | [diff] [blame] | 148 | /* <Output> */ |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 149 | /* sfnt :: The SFNT header. */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 150 | /* */ |
| 151 | /* <Return> */ |
Werner Lemberg | a929ba9 | 2000-06-25 06:47:11 +0000 | [diff] [blame] | 152 | /* FreeType error code. 0 means success. */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 153 | /* */ |
| 154 | /* <Note> */ |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 155 | /* The stream cursor must be at the font file's origin. */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 156 | /* */ |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 157 | /* This function recognizes fonts embedded in a `TrueType collection' */ |
| 158 | /* */ |
| 159 | /* The header will be checked whether it is valid by looking at the */ |
| 160 | /* values of `search_range', `entry_selector', and `range_shift'. */ |
David Turner | b1677a8 | 2000-05-29 20:37:41 +0000 | [diff] [blame] | 161 | /* */ |
David Turner | bc82f1b | 2002-03-01 02:26:22 +0000 | [diff] [blame] | 162 | FT_LOCAL_DEF( FT_Error ) |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 163 | TT_Load_SFNT_HeaderRec( TT_Face face, |
Werner Lemberg | 52005c3 | 2001-06-27 23:25:46 +0000 | [diff] [blame] | 164 | FT_Stream stream, |
| 165 | FT_Long face_index, |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 166 | SFNT_Header sfnt ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 167 | { |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 168 | FT_Error error; |
| 169 | FT_ULong format_tag; |
Werner Lemberg | 7a4fda8 | 2000-06-13 23:21:00 +0000 | [diff] [blame] | 170 | FT_Memory memory = stream->memory; |
Werner Lemberg | 920d41e | 2000-06-05 14:32:32 +0000 | [diff] [blame] | 171 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 172 | const FT_Frame_Field sfnt_header_fields[] = |
| 173 | { |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 174 | #undef FT_STRUCTURE |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 175 | #define FT_STRUCTURE SFNT_HeaderRec |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 176 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 177 | FT_FRAME_START( 8 ), |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 178 | FT_FRAME_USHORT( num_tables ), |
| 179 | FT_FRAME_USHORT( search_range ), |
| 180 | FT_FRAME_USHORT( entry_selector ), |
| 181 | FT_FRAME_USHORT( range_shift ), |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 182 | FT_FRAME_END |
| 183 | }; |
Werner Lemberg | 920d41e | 2000-06-05 14:32:32 +0000 | [diff] [blame] | 184 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 185 | const FT_Frame_Field ttc_header_fields[] = |
| 186 | { |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 187 | #undef FT_STRUCTURE |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 188 | #define FT_STRUCTURE TTC_HeaderRec |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 189 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 190 | FT_FRAME_START( 8 ), |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 191 | FT_FRAME_LONG( version ), |
| 192 | FT_FRAME_LONG( count ), |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 193 | FT_FRAME_END }; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 194 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 195 | |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 196 | FT_TRACE2(( "TT_Load_SFNT_HeaderRec: %08p, %ld\n", |
David Turner | 3581d06 | 2000-06-01 03:26:58 +0000 | [diff] [blame] | 197 | face, face_index )); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 198 | |
Werner Lemberg | e1bbc01 | 2000-07-19 06:25:56 +0000 | [diff] [blame] | 199 | face->ttc_header.tag = 0; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 200 | face->ttc_header.version = 0; |
Werner Lemberg | e1bbc01 | 2000-07-19 06:25:56 +0000 | [diff] [blame] | 201 | face->ttc_header.count = 0; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 202 | |
| 203 | face->num_tables = 0; |
| 204 | |
Werner Lemberg | fbeb41d | 2000-07-02 00:27:53 +0000 | [diff] [blame] | 205 | /* first of all, read the first 4 bytes. If it is `ttcf', then the */ |
| 206 | /* file is a TrueType collection, otherwise it can be any other */ |
| 207 | /* kind of font. */ |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 208 | if ( READ_ULong( format_tag ) ) |
| 209 | goto Exit; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 210 | |
David Turner | b1677a8 | 2000-05-29 20:37:41 +0000 | [diff] [blame] | 211 | if ( format_tag == TTAG_ttcf ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 212 | { |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 213 | FT_Int n; |
David Turner | e49ab25 | 2000-05-16 23:44:38 +0000 | [diff] [blame] | 214 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 215 | |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 216 | FT_TRACE3(( "TT_Load_SFNT_HeaderRec: file is a collection\n" )); |
David Turner | e49ab25 | 2000-05-16 23:44:38 +0000 | [diff] [blame] | 217 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 218 | /* it's a TrueType collection, i.e. a file containing several */ |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 219 | /* font files. Read the font directory now */ |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 220 | if ( FT_STREAM_READ_FIELDS( ttc_header_fields, &face->ttc_header ) ) |
David Turner | a56489e | 2000-02-13 13:41:56 +0000 | [diff] [blame] | 221 | goto Exit; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 222 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 223 | /* now read the offsets of each font in the file */ |
Werner Lemberg | e1bbc01 | 2000-07-19 06:25:56 +0000 | [diff] [blame] | 224 | if ( ALLOC_ARRAY( face->ttc_header.offsets, |
| 225 | face->ttc_header.count, |
| 226 | FT_ULong ) || |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 227 | FT_FRAME_ENTER( face->ttc_header.count * 4L ) ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 228 | goto Exit; |
| 229 | |
Werner Lemberg | e1bbc01 | 2000-07-19 06:25:56 +0000 | [diff] [blame] | 230 | for ( n = 0; n < face->ttc_header.count; n++ ) |
| 231 | face->ttc_header.offsets[n] = GET_ULong(); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 232 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 233 | FT_FRAME_EXIT(); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 234 | |
| 235 | /* check face index */ |
Werner Lemberg | e1bbc01 | 2000-07-19 06:25:56 +0000 | [diff] [blame] | 236 | if ( face_index >= face->ttc_header.count ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 237 | { |
Werner Lemberg | 1f7f0e8 | 2001-06-06 17:30:41 +0000 | [diff] [blame] | 238 | error = SFNT_Err_Bad_Argument; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 239 | goto Exit; |
David Turner | e49ab25 | 2000-05-16 23:44:38 +0000 | [diff] [blame] | 240 | } |
| 241 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 242 | /* seek to the appropriate TrueType file, then read tag */ |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 243 | if ( FT_STREAM_SEEK( face->ttc_header.offsets[face_index] ) || |
Werner Lemberg | e1bbc01 | 2000-07-19 06:25:56 +0000 | [diff] [blame] | 244 | READ_Long( format_tag ) ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 245 | goto Exit; |
| 246 | } |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 247 | |
David Turner | b1677a8 | 2000-05-29 20:37:41 +0000 | [diff] [blame] | 248 | /* the format tag was read, now check the rest of the header */ |
| 249 | sfnt->format_tag = format_tag; |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 250 | if ( FT_STREAM_READ_FIELDS( sfnt_header_fields, sfnt ) ) |
David Turner | b1677a8 | 2000-05-29 20:37:41 +0000 | [diff] [blame] | 251 | goto Exit; |
Werner Lemberg | 920d41e | 2000-06-05 14:32:32 +0000 | [diff] [blame] | 252 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 253 | /* now, check the values of `num_tables', `seach_range', etc. */ |
David Turner | b1677a8 | 2000-05-29 20:37:41 +0000 | [diff] [blame] | 254 | { |
Werner Lemberg | fbeb41d | 2000-07-02 00:27:53 +0000 | [diff] [blame] | 255 | FT_UInt num_tables = sfnt->num_tables; |
| 256 | FT_ULong entry_selector = 1L << sfnt->entry_selector; |
David Turner | 3581d06 | 2000-06-01 03:26:58 +0000 | [diff] [blame] | 257 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 258 | |
| 259 | /* IMPORTANT: Many fonts have an incorrect `search_range' value, so */ |
| 260 | /* we only check the `entry_selector' correctness here. */ |
| 261 | /* */ |
| 262 | if ( num_tables == 0 || |
| 263 | entry_selector > num_tables || |
| 264 | entry_selector * 2 <= num_tables ) |
David Turner | 3581d06 | 2000-06-01 03:26:58 +0000 | [diff] [blame] | 265 | { |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 266 | FT_TRACE2(( "TT_Load_SFNT_HeaderRec: file is not SFNT!\n" )); |
Werner Lemberg | 1f7f0e8 | 2001-06-06 17:30:41 +0000 | [diff] [blame] | 267 | error = SFNT_Err_Unknown_File_Format; |
David Turner | b1677a8 | 2000-05-29 20:37:41 +0000 | [diff] [blame] | 268 | } |
| 269 | } |
| 270 | |
David Turner | d42c68e | 2000-01-27 13:56:02 +0000 | [diff] [blame] | 271 | Exit: |
| 272 | return error; |
Werner Lemberg | 920d41e | 2000-06-05 14:32:32 +0000 | [diff] [blame] | 273 | } |
David Turner | b1677a8 | 2000-05-29 20:37:41 +0000 | [diff] [blame] | 274 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 275 | |
David Turner | d42c68e | 2000-01-27 13:56:02 +0000 | [diff] [blame] | 276 | /*************************************************************************/ |
| 277 | /* */ |
| 278 | /* <Function> */ |
| 279 | /* TT_Load_Directory */ |
| 280 | /* */ |
| 281 | /* <Description> */ |
| 282 | /* Loads the table directory into a face object. */ |
| 283 | /* */ |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 284 | /* <InOut> */ |
| 285 | /* face :: A handle to the target face object. */ |
| 286 | /* */ |
David Turner | d42c68e | 2000-01-27 13:56:02 +0000 | [diff] [blame] | 287 | /* <Input> */ |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 288 | /* stream :: The input stream. */ |
| 289 | /* sfnt :: The SFNT directory header. */ |
David Turner | d42c68e | 2000-01-27 13:56:02 +0000 | [diff] [blame] | 290 | /* */ |
| 291 | /* <Return> */ |
Werner Lemberg | a929ba9 | 2000-06-25 06:47:11 +0000 | [diff] [blame] | 292 | /* FreeType error code. 0 means success. */ |
David Turner | d42c68e | 2000-01-27 13:56:02 +0000 | [diff] [blame] | 293 | /* */ |
| 294 | /* <Note> */ |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 295 | /* The stream cursor must be at the font file's origin. */ |
David Turner | d42c68e | 2000-01-27 13:56:02 +0000 | [diff] [blame] | 296 | /* */ |
David Turner | bc82f1b | 2002-03-01 02:26:22 +0000 | [diff] [blame] | 297 | FT_LOCAL_DEF( FT_Error ) |
Werner Lemberg | 52005c3 | 2001-06-27 23:25:46 +0000 | [diff] [blame] | 298 | TT_Load_Directory( TT_Face face, |
| 299 | FT_Stream stream, |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 300 | SFNT_Header sfnt ) |
David Turner | d42c68e | 2000-01-27 13:56:02 +0000 | [diff] [blame] | 301 | { |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 302 | FT_Error error; |
David Turner | d42c68e | 2000-01-27 13:56:02 +0000 | [diff] [blame] | 303 | FT_Memory memory = stream->memory; |
David Turner | d42c68e | 2000-01-27 13:56:02 +0000 | [diff] [blame] | 304 | |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 305 | TT_TableRec *entry, *limit; |
David Turner | d42c68e | 2000-01-27 13:56:02 +0000 | [diff] [blame] | 306 | |
David Turner | d42c68e | 2000-01-27 13:56:02 +0000 | [diff] [blame] | 307 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 308 | FT_TRACE2(( "TT_Load_Directory: %08p\n", face )); |
| 309 | |
| 310 | FT_TRACE2(( "-- Tables count: %12u\n", sfnt->num_tables )); |
| 311 | FT_TRACE2(( "-- Format version: %08lx\n", sfnt->format_tag )); |
David Turner | d42c68e | 2000-01-27 13:56:02 +0000 | [diff] [blame] | 312 | |
David Turner | b1677a8 | 2000-05-29 20:37:41 +0000 | [diff] [blame] | 313 | face->num_tables = sfnt->num_tables; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 314 | |
| 315 | if ( ALLOC_ARRAY( face->dir_tables, |
| 316 | face->num_tables, |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 317 | TT_TableRec ) ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 318 | goto Exit; |
| 319 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 320 | if ( FT_FRAME_ENTER( face->num_tables * 16L ) ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 321 | goto Exit; |
| 322 | |
| 323 | entry = face->dir_tables; |
| 324 | limit = entry + face->num_tables; |
| 325 | |
| 326 | for ( ; entry < limit; entry++ ) |
David Turner | d42c68e | 2000-01-27 13:56:02 +0000 | [diff] [blame] | 327 | { /* loop through the tables and get all entries */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 328 | entry->Tag = GET_Tag4(); |
| 329 | entry->CheckSum = GET_ULong(); |
| 330 | entry->Offset = GET_Long(); |
| 331 | entry->Length = GET_Long(); |
| 332 | |
| 333 | FT_TRACE2(( " %c%c%c%c - %08lx - %08lx\n", |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 334 | (FT_Char)( entry->Tag >> 24 ), |
| 335 | (FT_Char)( entry->Tag >> 16 ), |
| 336 | (FT_Char)( entry->Tag >> 8 ), |
| 337 | (FT_Char)( entry->Tag ), |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 338 | entry->Offset, |
| 339 | entry->Length )); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 340 | } |
| 341 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 342 | FT_FRAME_EXIT(); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 343 | |
| 344 | FT_TRACE2(( "Directory loaded\n\n" )); |
| 345 | |
| 346 | Exit: |
| 347 | return error; |
| 348 | } |
| 349 | |
| 350 | |
| 351 | /*************************************************************************/ |
| 352 | /* */ |
| 353 | /* <Function> */ |
| 354 | /* TT_Load_Any */ |
| 355 | /* */ |
| 356 | /* <Description> */ |
Werner Lemberg | c8f9cf3 | 2000-07-19 02:59:31 +0000 | [diff] [blame] | 357 | /* Loads any font table into client memory. */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 358 | /* */ |
| 359 | /* <Input> */ |
| 360 | /* face :: The face object to look for. */ |
| 361 | /* */ |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 362 | /* tag :: The tag of table to load. Use the value 0 if you want */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 363 | /* to access the whole font file, else set this parameter */ |
| 364 | /* to a valid TrueType table tag that you can forge with */ |
| 365 | /* the MAKE_TT_TAG macro. */ |
| 366 | /* */ |
| 367 | /* offset :: The starting offset in the table (or the file if */ |
| 368 | /* tag == 0). */ |
| 369 | /* */ |
| 370 | /* length :: The address of the decision variable: */ |
| 371 | /* */ |
| 372 | /* If length == NULL: */ |
| 373 | /* Loads the whole table. Returns an error if */ |
| 374 | /* `offset' == 0! */ |
| 375 | /* */ |
| 376 | /* If *length == 0: */ |
| 377 | /* Exits immediately; returning the length of the given */ |
| 378 | /* table or of the font file, depending on the value of */ |
| 379 | /* `tag'. */ |
| 380 | /* */ |
| 381 | /* If *length != 0: */ |
| 382 | /* Loads the next `length' bytes of table or font, */ |
| 383 | /* starting at offset `offset' (in table or font too). */ |
| 384 | /* */ |
| 385 | /* <Output> */ |
| 386 | /* buffer :: The address of target buffer. */ |
| 387 | /* */ |
| 388 | /* <Return> */ |
Werner Lemberg | a929ba9 | 2000-06-25 06:47:11 +0000 | [diff] [blame] | 389 | /* FreeType error code. 0 means success. */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 390 | /* */ |
David Turner | bc82f1b | 2002-03-01 02:26:22 +0000 | [diff] [blame] | 391 | FT_LOCAL_DEF( FT_Error ) |
Werner Lemberg | 52005c3 | 2001-06-27 23:25:46 +0000 | [diff] [blame] | 392 | TT_Load_Any( TT_Face face, |
| 393 | FT_ULong tag, |
| 394 | FT_Long offset, |
| 395 | FT_Byte* buffer, |
| 396 | FT_ULong* length ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 397 | { |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 398 | FT_Error error; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 399 | FT_Stream stream; |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 400 | TT_Table table; |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 401 | FT_ULong size; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 402 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 403 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 404 | if ( tag != 0 ) |
| 405 | { |
| 406 | /* look for tag in font directory */ |
| 407 | table = TT_LookUp_Table( face, tag ); |
| 408 | if ( !table ) |
| 409 | { |
Werner Lemberg | 1f7f0e8 | 2001-06-06 17:30:41 +0000 | [diff] [blame] | 410 | error = SFNT_Err_Table_Missing; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 411 | goto Exit; |
| 412 | } |
| 413 | |
| 414 | offset += table->Offset; |
| 415 | size = table->Length; |
| 416 | } |
| 417 | else |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 418 | /* tag == 0 -- the user wants to access the font file directly */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 419 | size = face->root.stream->size; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 420 | |
| 421 | if ( length && *length == 0 ) |
| 422 | { |
| 423 | *length = size; |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 424 | |
Werner Lemberg | 1f7f0e8 | 2001-06-06 17:30:41 +0000 | [diff] [blame] | 425 | return SFNT_Err_Ok; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 426 | } |
| 427 | |
| 428 | if ( length ) |
| 429 | size = *length; |
| 430 | |
| 431 | stream = face->root.stream; |
Werner Lemberg | b1dd353 | 2000-07-31 22:51:00 +0000 | [diff] [blame] | 432 | /* the `if' is syntactic sugar for picky compilers */ |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 433 | if ( FT_STREAM_READ_AT( offset, buffer, size ) ) |
David Turner | 24d7024 | 2000-08-17 01:09:06 +0000 | [diff] [blame] | 434 | goto Exit; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 435 | |
| 436 | Exit: |
| 437 | return error; |
| 438 | } |
| 439 | |
| 440 | |
| 441 | /*************************************************************************/ |
| 442 | /* */ |
| 443 | /* <Function> */ |
Tom Kacvinsky | 200a1a4 | 2000-11-13 07:31:21 +0000 | [diff] [blame] | 444 | /* TT_Load_Generic_Header */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 445 | /* */ |
| 446 | /* <Description> */ |
Tom Kacvinsky | 200a1a4 | 2000-11-13 07:31:21 +0000 | [diff] [blame] | 447 | /* Loads the TrueType table `head' or `bhed'. */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 448 | /* */ |
| 449 | /* <Input> */ |
| 450 | /* face :: A handle to the target face object. */ |
| 451 | /* stream :: The input stream. */ |
| 452 | /* */ |
| 453 | /* <Return> */ |
Werner Lemberg | a929ba9 | 2000-06-25 06:47:11 +0000 | [diff] [blame] | 454 | /* FreeType error code. 0 means success. */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 455 | /* */ |
Werner Lemberg | 52005c3 | 2001-06-27 23:25:46 +0000 | [diff] [blame] | 456 | static FT_Error |
| 457 | TT_Load_Generic_Header( TT_Face face, |
| 458 | FT_Stream stream, |
| 459 | FT_ULong tag ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 460 | { |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 461 | FT_Error error; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 462 | TT_Header* header; |
David Turner | b1677a8 | 2000-05-29 20:37:41 +0000 | [diff] [blame] | 463 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 464 | static const FT_Frame_Field header_fields[] = |
| 465 | { |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 466 | #undef FT_STRUCTURE |
| 467 | #define FT_STRUCTURE TT_Header |
| 468 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 469 | FT_FRAME_START( 54 ), |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 470 | FT_FRAME_ULONG ( Table_Version ), |
| 471 | FT_FRAME_ULONG ( Font_Revision ), |
| 472 | FT_FRAME_LONG ( CheckSum_Adjust ), |
| 473 | FT_FRAME_LONG ( Magic_Number ), |
| 474 | FT_FRAME_USHORT( Flags ), |
| 475 | FT_FRAME_USHORT( Units_Per_EM ), |
| 476 | FT_FRAME_LONG ( Created[0] ), |
| 477 | FT_FRAME_LONG ( Created[1] ), |
| 478 | FT_FRAME_LONG ( Modified[0] ), |
| 479 | FT_FRAME_LONG ( Modified[1] ), |
| 480 | FT_FRAME_SHORT ( xMin ), |
| 481 | FT_FRAME_SHORT ( yMin ), |
| 482 | FT_FRAME_SHORT ( xMax ), |
| 483 | FT_FRAME_SHORT ( yMax ), |
| 484 | FT_FRAME_USHORT( Mac_Style ), |
| 485 | FT_FRAME_USHORT( Lowest_Rec_PPEM ), |
| 486 | FT_FRAME_SHORT ( Font_Direction ), |
| 487 | FT_FRAME_SHORT ( Index_To_Loc_Format ), |
| 488 | FT_FRAME_SHORT ( Glyph_Data_Format ), |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 489 | FT_FRAME_END |
| 490 | }; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 491 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 492 | |
Werner Lemberg | 34e5270 | 2000-11-13 16:58:01 +0000 | [diff] [blame] | 493 | FT_TRACE2(( "TT_Load_Generic_Header: " |
| 494 | "%08p, looking up font table `%c%c%c%c'.\n", |
| 495 | face, |
| 496 | (FT_Char)( tag >> 24 ), |
| 497 | (FT_Char)( tag >> 16 ), |
| 498 | (FT_Char)( tag >> 8 ), |
| 499 | (FT_Char)( tag ) )); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 500 | |
David Turner | f96594f | 2000-11-06 23:07:51 +0000 | [diff] [blame] | 501 | error = face->goto_table( face, tag, stream, 0 ); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 502 | if ( error ) |
| 503 | { |
Werner Lemberg | 04bfe31 | 2001-12-11 14:09:06 +0000 | [diff] [blame] | 504 | FT_TRACE2(( "TT_Load_Generic_Header: Font table is missing!\n" )); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 505 | goto Exit; |
| 506 | } |
| 507 | |
David Turner | a56489e | 2000-02-13 13:41:56 +0000 | [diff] [blame] | 508 | header = &face->header; |
| 509 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 510 | if ( FT_STREAM_READ_FIELDS( header_fields, header ) ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 511 | goto Exit; |
| 512 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 513 | FT_TRACE2(( " Units per EM: %8u\n", header->Units_Per_EM )); |
| 514 | FT_TRACE2(( " IndexToLoc: %8d\n", header->Index_To_Loc_Format )); |
Werner Lemberg | 04bfe31 | 2001-12-11 14:09:06 +0000 | [diff] [blame] | 515 | FT_TRACE2(( "TT_Load_Generic_Header: Font table loaded.\n" )); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 516 | |
| 517 | Exit: |
| 518 | return error; |
| 519 | } |
| 520 | |
Werner Lemberg | 4b68007 | 2000-11-07 06:30:29 +0000 | [diff] [blame] | 521 | |
David Turner | bc82f1b | 2002-03-01 02:26:22 +0000 | [diff] [blame] | 522 | FT_LOCAL_DEF( FT_Error ) |
Werner Lemberg | 52005c3 | 2001-06-27 23:25:46 +0000 | [diff] [blame] | 523 | TT_Load_Header( TT_Face face, |
| 524 | FT_Stream stream ) |
David Turner | f96594f | 2000-11-06 23:07:51 +0000 | [diff] [blame] | 525 | { |
| 526 | return TT_Load_Generic_Header( face, stream, TTAG_head ); |
| 527 | } |
| 528 | |
Werner Lemberg | 4b68007 | 2000-11-07 06:30:29 +0000 | [diff] [blame] | 529 | |
David Turner | f96594f | 2000-11-06 23:07:51 +0000 | [diff] [blame] | 530 | #ifdef TT_CONFIG_OPTION_EMBEDDED_BITMAPS |
Werner Lemberg | 4b68007 | 2000-11-07 06:30:29 +0000 | [diff] [blame] | 531 | |
David Turner | bc82f1b | 2002-03-01 02:26:22 +0000 | [diff] [blame] | 532 | FT_LOCAL_DEF( FT_Error ) |
Werner Lemberg | 52005c3 | 2001-06-27 23:25:46 +0000 | [diff] [blame] | 533 | TT_Load_Bitmap_Header( TT_Face face, |
| 534 | FT_Stream stream ) |
David Turner | f96594f | 2000-11-06 23:07:51 +0000 | [diff] [blame] | 535 | { |
| 536 | return TT_Load_Generic_Header( face, stream, TTAG_bhed ); |
| 537 | } |
Werner Lemberg | 4b68007 | 2000-11-07 06:30:29 +0000 | [diff] [blame] | 538 | |
| 539 | #endif /* TT_CONFIG_OPTION_EMBEDDED_BITMAPS */ |
| 540 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 541 | |
| 542 | /*************************************************************************/ |
| 543 | /* */ |
| 544 | /* <Function> */ |
| 545 | /* TT_Load_MaxProfile */ |
| 546 | /* */ |
| 547 | /* <Description> */ |
| 548 | /* Loads the maximum profile into a face object. */ |
| 549 | /* */ |
| 550 | /* <Input> */ |
| 551 | /* face :: A handle to the target face object. */ |
| 552 | /* stream :: The input stream. */ |
| 553 | /* */ |
| 554 | /* <Return> */ |
Werner Lemberg | a929ba9 | 2000-06-25 06:47:11 +0000 | [diff] [blame] | 555 | /* FreeType error code. 0 means success. */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 556 | /* */ |
David Turner | bc82f1b | 2002-03-01 02:26:22 +0000 | [diff] [blame] | 557 | FT_LOCAL_DEF( FT_Error ) |
Werner Lemberg | 52005c3 | 2001-06-27 23:25:46 +0000 | [diff] [blame] | 558 | TT_Load_MaxProfile( TT_Face face, |
| 559 | FT_Stream stream ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 560 | { |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 561 | FT_Error error; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 562 | TT_MaxProfile* maxProfile = &face->max_profile; |
David Turner | b1677a8 | 2000-05-29 20:37:41 +0000 | [diff] [blame] | 563 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 564 | const FT_Frame_Field maxp_fields[] = |
| 565 | { |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 566 | #undef FT_STRUCTURE |
| 567 | #define FT_STRUCTURE TT_MaxProfile |
| 568 | |
Werner Lemberg | 36af3ea | 2001-07-17 12:37:54 +0000 | [diff] [blame] | 569 | FT_FRAME_START( 6 ), |
| 570 | FT_FRAME_LONG ( version ), |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 571 | FT_FRAME_USHORT( numGlyphs ), |
Werner Lemberg | 36af3ea | 2001-07-17 12:37:54 +0000 | [diff] [blame] | 572 | FT_FRAME_END |
| 573 | }; |
| 574 | |
| 575 | const FT_Frame_Field maxp_fields_extra[] = |
| 576 | { |
| 577 | FT_FRAME_START( 26 ), |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 578 | FT_FRAME_USHORT( maxPoints ), |
| 579 | FT_FRAME_USHORT( maxContours ), |
| 580 | FT_FRAME_USHORT( maxCompositePoints ), |
| 581 | FT_FRAME_USHORT( maxCompositeContours ), |
| 582 | FT_FRAME_USHORT( maxZones ), |
| 583 | FT_FRAME_USHORT( maxTwilightPoints ), |
| 584 | FT_FRAME_USHORT( maxStorage ), |
| 585 | FT_FRAME_USHORT( maxFunctionDefs ), |
| 586 | FT_FRAME_USHORT( maxInstructionDefs ), |
| 587 | FT_FRAME_USHORT( maxStackElements ), |
| 588 | FT_FRAME_USHORT( maxSizeOfInstructions ), |
| 589 | FT_FRAME_USHORT( maxComponentElements ), |
| 590 | FT_FRAME_USHORT( maxComponentDepth ), |
Werner Lemberg | 36af3ea | 2001-07-17 12:37:54 +0000 | [diff] [blame] | 591 | FT_FRAME_END |
| 592 | }; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 593 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 594 | |
| 595 | FT_TRACE2(( "Load_TT_MaxProfile: %08p\n", face )); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 596 | |
| 597 | error = face->goto_table( face, TTAG_maxp, stream, 0 ); |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 598 | if ( error ) |
| 599 | goto Exit; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 600 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 601 | if ( FT_STREAM_READ_FIELDS( maxp_fields, maxProfile ) ) |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 602 | goto Exit; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 603 | |
Werner Lemberg | 36af3ea | 2001-07-17 12:37:54 +0000 | [diff] [blame] | 604 | maxProfile->maxPoints = 0; |
| 605 | maxProfile->maxContours = 0; |
| 606 | maxProfile->maxCompositePoints = 0; |
| 607 | maxProfile->maxCompositeContours = 0; |
| 608 | maxProfile->maxZones = 0; |
| 609 | maxProfile->maxTwilightPoints = 0; |
| 610 | maxProfile->maxStorage = 0; |
| 611 | maxProfile->maxFunctionDefs = 0; |
| 612 | maxProfile->maxInstructionDefs = 0; |
| 613 | maxProfile->maxStackElements = 0; |
| 614 | maxProfile->maxSizeOfInstructions = 0; |
| 615 | maxProfile->maxComponentElements = 0; |
| 616 | maxProfile->maxComponentDepth = 0; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 617 | |
Werner Lemberg | 36af3ea | 2001-07-17 12:37:54 +0000 | [diff] [blame] | 618 | if ( maxProfile->version >= 0x10000L ) |
| 619 | { |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 620 | if ( FT_STREAM_READ_FIELDS( maxp_fields_extra, maxProfile ) ) |
Werner Lemberg | 36af3ea | 2001-07-17 12:37:54 +0000 | [diff] [blame] | 621 | goto Exit; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 622 | |
Werner Lemberg | 36af3ea | 2001-07-17 12:37:54 +0000 | [diff] [blame] | 623 | /* XXX: an adjustment that is necessary to load certain */ |
| 624 | /* broken fonts like `Keystrokes MT' :-( */ |
| 625 | /* */ |
| 626 | /* We allocate 64 function entries by default when */ |
| 627 | /* the maxFunctionDefs field is null. */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 628 | |
Werner Lemberg | 36af3ea | 2001-07-17 12:37:54 +0000 | [diff] [blame] | 629 | if ( maxProfile->maxFunctionDefs == 0 ) |
| 630 | maxProfile->maxFunctionDefs = 64; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 631 | |
Werner Lemberg | 36af3ea | 2001-07-17 12:37:54 +0000 | [diff] [blame] | 632 | face->root.num_glyphs = maxProfile->numGlyphs; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 633 | |
Werner Lemberg | 36af3ea | 2001-07-17 12:37:54 +0000 | [diff] [blame] | 634 | face->root.internal->max_points = |
| 635 | (FT_UShort)MAX( maxProfile->maxCompositePoints, |
| 636 | maxProfile->maxPoints ); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 637 | |
Werner Lemberg | 36af3ea | 2001-07-17 12:37:54 +0000 | [diff] [blame] | 638 | face->root.internal->max_contours = |
| 639 | (FT_Short)MAX( maxProfile->maxCompositeContours, |
| 640 | maxProfile->maxContours ); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 641 | |
Werner Lemberg | 36af3ea | 2001-07-17 12:37:54 +0000 | [diff] [blame] | 642 | face->max_components = (FT_ULong)maxProfile->maxComponentElements + |
| 643 | maxProfile->maxComponentDepth; |
| 644 | |
| 645 | /* XXX: some fonts have maxComponents set to 0; we will */ |
| 646 | /* then use 16 of them by default. */ |
| 647 | if ( face->max_components == 0 ) |
| 648 | face->max_components = 16; |
| 649 | |
| 650 | /* We also increase maxPoints and maxContours in order to support */ |
| 651 | /* some broken fonts. */ |
Werner Lemberg | 3975e2e | 2002-01-09 21:01:18 +0000 | [diff] [blame] | 652 | face->root.internal->max_points += (FT_UShort)8; |
| 653 | face->root.internal->max_contours += (FT_Short) 4; |
Werner Lemberg | 36af3ea | 2001-07-17 12:37:54 +0000 | [diff] [blame] | 654 | } |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 655 | |
| 656 | FT_TRACE2(( "MAXP loaded.\n" )); |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 657 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 658 | Exit: |
| 659 | return error; |
| 660 | } |
| 661 | |
| 662 | |
| 663 | /*************************************************************************/ |
| 664 | /* */ |
| 665 | /* <Function> */ |
| 666 | /* TT_Load_Metrics */ |
| 667 | /* */ |
| 668 | /* <Description> */ |
| 669 | /* Loads the horizontal or vertical metrics table into a face object. */ |
| 670 | /* */ |
| 671 | /* <Input> */ |
| 672 | /* face :: A handle to the target face object. */ |
| 673 | /* stream :: The input stream. */ |
| 674 | /* vertical :: A boolean flag. If set, load vertical metrics. */ |
| 675 | /* */ |
| 676 | /* <Return> */ |
Werner Lemberg | a929ba9 | 2000-06-25 06:47:11 +0000 | [diff] [blame] | 677 | /* FreeType error code. 0 means success. */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 678 | /* */ |
Werner Lemberg | 52005c3 | 2001-06-27 23:25:46 +0000 | [diff] [blame] | 679 | static FT_Error |
| 680 | TT_Load_Metrics( TT_Face face, |
| 681 | FT_Stream stream, |
| 682 | FT_Bool vertical ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 683 | { |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 684 | FT_Error error; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 685 | FT_Memory memory = stream->memory; |
| 686 | |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 687 | FT_ULong table_len; |
| 688 | FT_Long num_shorts, num_longs, num_shorts_checked; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 689 | |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 690 | TT_LongMetrics * longs; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 691 | TT_ShortMetrics** shorts; |
| 692 | |
Werner Lemberg | fbeb41d | 2000-07-02 00:27:53 +0000 | [diff] [blame] | 693 | |
| 694 | FT_TRACE2(( "TT_Load_%s_Metrics: %08p\n", vertical ? "Vertical" |
| 695 | : "Horizontal", |
| 696 | face )); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 697 | |
| 698 | if ( vertical ) |
| 699 | { |
| 700 | /* The table is optional, quit silently if it wasn't found */ |
| 701 | /* XXX: Some fonts have a valid vertical header with a non-null */ |
| 702 | /* `number_of_VMetrics' fields, but no corresponding `vmtx' */ |
| 703 | /* table to get the metrics from (e.g. mingliu). */ |
| 704 | /* */ |
| 705 | /* For safety, we set the field to 0! */ |
| 706 | /* */ |
| 707 | error = face->goto_table( face, TTAG_vmtx, stream, &table_len ); |
| 708 | if ( error ) |
| 709 | { |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 710 | /* Set number_Of_VMetrics to 0! */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 711 | FT_TRACE2(( " no vertical header in file.\n" )); |
| 712 | face->vertical.number_Of_VMetrics = 0; |
Werner Lemberg | 1f7f0e8 | 2001-06-06 17:30:41 +0000 | [diff] [blame] | 713 | error = SFNT_Err_Ok; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 714 | goto Exit; |
| 715 | } |
| 716 | |
| 717 | num_longs = face->vertical.number_Of_VMetrics; |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 718 | longs = (TT_LongMetrics *)&face->vertical.long_metrics; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 719 | shorts = (TT_ShortMetrics**)&face->vertical.short_metrics; |
| 720 | } |
| 721 | else |
| 722 | { |
| 723 | error = face->goto_table( face, TTAG_hmtx, stream, &table_len ); |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 724 | if ( error ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 725 | { |
| 726 | FT_ERROR(( " no horizontal metrics in file!\n" )); |
Werner Lemberg | 1f7f0e8 | 2001-06-06 17:30:41 +0000 | [diff] [blame] | 727 | error = SFNT_Err_Hmtx_Table_Missing; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 728 | goto Exit; |
| 729 | } |
| 730 | |
| 731 | num_longs = face->horizontal.number_Of_HMetrics; |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 732 | longs = (TT_LongMetrics *)&face->horizontal.long_metrics; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 733 | shorts = (TT_ShortMetrics**)&face->horizontal.short_metrics; |
| 734 | } |
| 735 | |
| 736 | /* never trust derived values */ |
| 737 | |
| 738 | num_shorts = face->max_profile.numGlyphs - num_longs; |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 739 | num_shorts_checked = ( table_len - num_longs * 4L ) / 2; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 740 | |
| 741 | if ( num_shorts < 0 ) |
| 742 | { |
Werner Lemberg | fbeb41d | 2000-07-02 00:27:53 +0000 | [diff] [blame] | 743 | FT_ERROR(( "TT_Load_%s_Metrics: more metrics than glyphs!\n", |
| 744 | vertical ? "Vertical" |
| 745 | : "Horizontal" )); |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 746 | |
Werner Lemberg | 1f7f0e8 | 2001-06-06 17:30:41 +0000 | [diff] [blame] | 747 | error = vertical ? SFNT_Err_Invalid_Vert_Metrics |
| 748 | : SFNT_Err_Invalid_Horiz_Metrics; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 749 | goto Exit; |
| 750 | } |
| 751 | |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 752 | if ( ALLOC_ARRAY( *longs, num_longs, TT_LongMetricsRec ) || |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 753 | ALLOC_ARRAY( *shorts, num_shorts, TT_ShortMetrics ) ) |
| 754 | goto Exit; |
| 755 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 756 | if ( FT_FRAME_ENTER( table_len ) ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 757 | goto Exit; |
| 758 | |
| 759 | { |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 760 | TT_LongMetrics cur = *longs; |
| 761 | TT_LongMetrics limit = cur + num_longs; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 762 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 763 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 764 | for ( ; cur < limit; cur++ ) |
| 765 | { |
| 766 | cur->advance = GET_UShort(); |
| 767 | cur->bearing = GET_Short(); |
| 768 | } |
| 769 | } |
| 770 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 771 | /* do we have an inconsistent number of metric values? */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 772 | { |
| 773 | TT_ShortMetrics* cur = *shorts; |
| 774 | TT_ShortMetrics* limit = cur + MIN( num_shorts, num_shorts_checked ); |
| 775 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 776 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 777 | for ( ; cur < limit; cur++ ) |
| 778 | *cur = GET_Short(); |
| 779 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 780 | /* we fill up the missing left side bearings with the */ |
| 781 | /* last valid value. Since this will occur for buggy CJK */ |
| 782 | /* fonts usually only, nothing serious will happen */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 783 | if ( num_shorts > num_shorts_checked && num_shorts_checked > 0 ) |
| 784 | { |
Tom Kacvinsky | 4a1bc8f | 2001-04-02 17:47:16 +0000 | [diff] [blame] | 785 | FT_Short val = (*shorts)[num_shorts_checked - 1]; |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 786 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 787 | |
| 788 | limit = *shorts + num_shorts; |
| 789 | for ( ; cur < limit; cur++ ) |
| 790 | *cur = val; |
| 791 | } |
| 792 | } |
| 793 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 794 | FT_FRAME_EXIT(); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 795 | |
| 796 | FT_TRACE2(( "loaded\n" )); |
| 797 | |
| 798 | Exit: |
| 799 | return error; |
| 800 | } |
| 801 | |
| 802 | |
| 803 | /*************************************************************************/ |
| 804 | /* */ |
| 805 | /* <Function> */ |
| 806 | /* TT_Load_Metrics_Header */ |
| 807 | /* */ |
| 808 | /* <Description> */ |
| 809 | /* Loads the horizontal or vertical header in a face object. */ |
| 810 | /* */ |
| 811 | /* <Input> */ |
| 812 | /* face :: A handle to the target face object. */ |
| 813 | /* stream :: The input stream. */ |
| 814 | /* vertical :: A boolean flag. If set, load vertical metrics. */ |
| 815 | /* */ |
| 816 | /* <Return> */ |
Werner Lemberg | a929ba9 | 2000-06-25 06:47:11 +0000 | [diff] [blame] | 817 | /* FreeType error code. 0 means success. */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 818 | /* */ |
David Turner | bc82f1b | 2002-03-01 02:26:22 +0000 | [diff] [blame] | 819 | FT_LOCAL_DEF( FT_Error ) |
Werner Lemberg | 52005c3 | 2001-06-27 23:25:46 +0000 | [diff] [blame] | 820 | TT_Load_Metrics_Header( TT_Face face, |
| 821 | FT_Stream stream, |
| 822 | FT_Bool vertical ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 823 | { |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 824 | FT_Error error; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 825 | TT_HoriHeader* header; |
David Turner | b1677a8 | 2000-05-29 20:37:41 +0000 | [diff] [blame] | 826 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 827 | const FT_Frame_Field metrics_header_fields[] = |
| 828 | { |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 829 | #undef FT_STRUCTURE |
| 830 | #define FT_STRUCTURE TT_HoriHeader |
| 831 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 832 | FT_FRAME_START( 36 ), |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 833 | FT_FRAME_ULONG ( Version ), |
| 834 | FT_FRAME_SHORT ( Ascender ), |
| 835 | FT_FRAME_SHORT ( Descender ), |
| 836 | FT_FRAME_SHORT ( Line_Gap ), |
| 837 | FT_FRAME_USHORT( advance_Width_Max ), |
| 838 | FT_FRAME_SHORT ( min_Left_Side_Bearing ), |
| 839 | FT_FRAME_SHORT ( min_Right_Side_Bearing ), |
| 840 | FT_FRAME_SHORT ( xMax_Extent ), |
| 841 | FT_FRAME_SHORT ( caret_Slope_Rise ), |
| 842 | FT_FRAME_SHORT ( caret_Slope_Run ), |
Werner Lemberg | 5dea444 | 2001-08-13 11:44:29 +0000 | [diff] [blame] | 843 | FT_FRAME_SHORT ( caret_Offset ), |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 844 | FT_FRAME_SHORT ( Reserved[0] ), |
| 845 | FT_FRAME_SHORT ( Reserved[1] ), |
| 846 | FT_FRAME_SHORT ( Reserved[2] ), |
| 847 | FT_FRAME_SHORT ( Reserved[3] ), |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 848 | FT_FRAME_SHORT ( metric_Data_Format ), |
| 849 | FT_FRAME_USHORT( number_Of_HMetrics ), |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 850 | FT_FRAME_END |
| 851 | }; |
| 852 | |
David Turner | b1677a8 | 2000-05-29 20:37:41 +0000 | [diff] [blame] | 853 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 854 | FT_TRACE2(( vertical ? "Vertical header " : "Horizontal header " )); |
| 855 | |
| 856 | if ( vertical ) |
| 857 | { |
| 858 | face->vertical_info = 0; |
| 859 | |
| 860 | /* The vertical header table is optional, so return quietly if */ |
| 861 | /* we don't find it. */ |
| 862 | error = face->goto_table( face, TTAG_vhea, stream, 0 ); |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 863 | if ( error ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 864 | { |
Werner Lemberg | 1f7f0e8 | 2001-06-06 17:30:41 +0000 | [diff] [blame] | 865 | error = SFNT_Err_Ok; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 866 | goto Exit; |
| 867 | } |
| 868 | |
| 869 | face->vertical_info = 1; |
| 870 | header = (TT_HoriHeader*)&face->vertical; |
| 871 | } |
| 872 | else |
| 873 | { |
Werner Lemberg | fbeb41d | 2000-07-02 00:27:53 +0000 | [diff] [blame] | 874 | /* The horizontal header is mandatory; return an error if we */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 875 | /* don't find it. */ |
| 876 | error = face->goto_table( face, TTAG_hhea, stream, 0 ); |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 877 | if ( error ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 878 | { |
Werner Lemberg | 1f7f0e8 | 2001-06-06 17:30:41 +0000 | [diff] [blame] | 879 | error = SFNT_Err_Horiz_Header_Missing; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 880 | goto Exit; |
| 881 | } |
| 882 | |
| 883 | header = &face->horizontal; |
| 884 | } |
| 885 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 886 | if ( FT_STREAM_READ_FIELDS( metrics_header_fields, header ) ) |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 887 | goto Exit; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 888 | |
| 889 | header->long_metrics = NULL; |
| 890 | header->short_metrics = NULL; |
| 891 | |
| 892 | FT_TRACE2(( "loaded\n" )); |
| 893 | |
| 894 | /* Now try to load the corresponding metrics */ |
| 895 | |
| 896 | error = TT_Load_Metrics( face, stream, vertical ); |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 897 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 898 | Exit: |
| 899 | return error; |
| 900 | } |
| 901 | |
| 902 | |
| 903 | /*************************************************************************/ |
| 904 | /* */ |
| 905 | /* <Function> */ |
| 906 | /* TT_Load_Names */ |
| 907 | /* */ |
| 908 | /* <Description> */ |
| 909 | /* Loads the name records. */ |
| 910 | /* */ |
| 911 | /* <Input> */ |
| 912 | /* face :: A handle to the target face object. */ |
| 913 | /* stream :: The input stream. */ |
| 914 | /* */ |
| 915 | /* <Return> */ |
Werner Lemberg | a929ba9 | 2000-06-25 06:47:11 +0000 | [diff] [blame] | 916 | /* FreeType error code. 0 means success. */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 917 | /* */ |
David Turner | bc82f1b | 2002-03-01 02:26:22 +0000 | [diff] [blame] | 918 | FT_LOCAL_DEF( FT_Error ) |
Werner Lemberg | 52005c3 | 2001-06-27 23:25:46 +0000 | [diff] [blame] | 919 | TT_Load_Names( TT_Face face, |
| 920 | FT_Stream stream ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 921 | { |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 922 | FT_Error error; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 923 | FT_Memory memory = stream->memory; |
| 924 | |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 925 | FT_ULong table_pos, table_len; |
David Turner | c03d9cf | 2002-02-27 23:10:19 +0000 | [diff] [blame] | 926 | FT_ULong storageOffset, storageSize; |
| 927 | FT_Byte* storage; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 928 | |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 929 | TT_NameTable names; |
David Turner | b1677a8 | 2000-05-29 20:37:41 +0000 | [diff] [blame] | 930 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 931 | const FT_Frame_Field name_table_fields[] = |
| 932 | { |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 933 | #undef FT_STRUCTURE |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 934 | #define FT_STRUCTURE TT_NameTableRec |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 935 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 936 | FT_FRAME_START( 6 ), |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 937 | FT_FRAME_USHORT( format ), |
| 938 | FT_FRAME_USHORT( numNameRecords ), |
| 939 | FT_FRAME_USHORT( storageOffset ), |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 940 | FT_FRAME_END |
| 941 | }; |
David Turner | a56489e | 2000-02-13 13:41:56 +0000 | [diff] [blame] | 942 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 943 | const FT_Frame_Field name_record_fields[] = |
| 944 | { |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 945 | #undef FT_STRUCTURE |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 946 | #define FT_STRUCTURE TT_NameEntryRec |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 947 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 948 | /* no FT_FRAME_START */ |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 949 | FT_FRAME_USHORT( platformID ), |
| 950 | FT_FRAME_USHORT( encodingID ), |
| 951 | FT_FRAME_USHORT( languageID ), |
| 952 | FT_FRAME_USHORT( nameID ), |
| 953 | FT_FRAME_USHORT( stringLength ), |
| 954 | FT_FRAME_USHORT( stringOffset ), |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 955 | FT_FRAME_END |
| 956 | }; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 957 | |
| 958 | |
| 959 | FT_TRACE2(( "Names " )); |
| 960 | |
| 961 | error = face->goto_table( face, TTAG_name, stream, &table_len ); |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 962 | if ( error ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 963 | { |
| 964 | /* The name table is required so indicate failure. */ |
| 965 | FT_TRACE2(( "is missing!\n" )); |
Werner Lemberg | 1f7f0e8 | 2001-06-06 17:30:41 +0000 | [diff] [blame] | 966 | error = SFNT_Err_Name_Table_Missing; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 967 | goto Exit; |
| 968 | } |
| 969 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 970 | table_pos = FT_STREAM_POS(); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 971 | |
David Turner | a56489e | 2000-02-13 13:41:56 +0000 | [diff] [blame] | 972 | names = &face->name_table; |
| 973 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 974 | if ( FT_STREAM_READ_FIELDS( name_table_fields, names ) ) |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 975 | goto Exit; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 976 | |
David Turner | c03d9cf | 2002-02-27 23:10:19 +0000 | [diff] [blame] | 977 | /* check the 'storageOffset' field */ |
| 978 | storageOffset = names->storageOffset; |
| 979 | if ( storageOffset < (FT_ULong)(6 + 12*names->numNameRecords) || |
| 980 | table_len <= storageOffset ) |
| 981 | { |
| 982 | FT_ERROR(( "TT.load_names: invalid 'name' table\n" )); |
| 983 | error = SFNT_Err_Name_Table_Missing; |
| 984 | goto Exit; |
| 985 | } |
| 986 | |
| 987 | storageSize = (FT_ULong)(table_len - storageOffset); |
| 988 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 989 | /* Allocate the array of name records. */ |
David Turner | c03d9cf | 2002-02-27 23:10:19 +0000 | [diff] [blame] | 990 | if ( ALLOC( names->names, |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 991 | names->numNameRecords*sizeof(TT_NameEntryRec) + storageSize ) || |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 992 | FT_FRAME_ENTER( names->numNameRecords * 12L ) ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 993 | goto Exit; |
| 994 | |
David Turner | c03d9cf | 2002-02-27 23:10:19 +0000 | [diff] [blame] | 995 | storage = (FT_Byte*)(names->names + names->numNameRecords); |
| 996 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 997 | /* Load the name records and determine how much storage is needed */ |
| 998 | /* to hold the strings themselves. */ |
| 999 | { |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 1000 | TT_NameEntryRec* cur = names->names; |
| 1001 | TT_NameEntryRec* limit = cur + names->numNameRecords; |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1002 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1003 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1004 | for ( ; cur < limit; cur ++ ) |
| 1005 | { |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1006 | if ( FT_STREAM_READ_FIELDS( name_record_fields, cur ) ) |
Werner Lemberg | b1dd353 | 2000-07-31 22:51:00 +0000 | [diff] [blame] | 1007 | break; |
David Turner | b1677a8 | 2000-05-29 20:37:41 +0000 | [diff] [blame] | 1008 | |
David Turner | c03d9cf | 2002-02-27 23:10:19 +0000 | [diff] [blame] | 1009 | /* invalid name entries will have "cur->string" set to NULL !! */ |
| 1010 | if ( (FT_ULong)(cur->stringOffset + cur->stringLength) < storageSize ) |
| 1011 | cur->string = storage + cur->stringOffset; |
David Turner | 28ea6f6 | 2002-03-05 16:12:57 +0000 | [diff] [blame] | 1012 | else |
| 1013 | { |
| 1014 | /* that's an invalid entry */ |
| 1015 | cur->stringOffset = 0; |
| 1016 | cur->string = NULL; |
| 1017 | } |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1018 | } |
| 1019 | } |
| 1020 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1021 | FT_FRAME_EXIT(); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1022 | |
David Turner | c03d9cf | 2002-02-27 23:10:19 +0000 | [diff] [blame] | 1023 | if (error) |
| 1024 | goto Exit; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1025 | |
David Turner | 28ea6f6 | 2002-03-05 16:12:57 +0000 | [diff] [blame] | 1026 | storageOffset -= 6 + 12*names->numNameRecords; |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1027 | if ( FT_STREAM_SKIP( storageOffset ) || |
| 1028 | FT_STREAM_READ( storage, storageSize ) ) |
David Turner | 28ea6f6 | 2002-03-05 16:12:57 +0000 | [diff] [blame] | 1029 | goto Exit; |
| 1030 | |
| 1031 | |
David Turner | e49ab25 | 2000-05-16 23:44:38 +0000 | [diff] [blame] | 1032 | #ifdef FT_DEBUG_LEVEL_TRACE |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1033 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1034 | /* Print Name Record Table in case of debugging */ |
| 1035 | { |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 1036 | TT_NameEntryRec* cur = names->names; |
| 1037 | TT_NameEntryRec* limit = cur + names->numNameRecords; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1038 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1039 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1040 | for ( ; cur < limit; cur++ ) |
| 1041 | { |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 1042 | FT_UInt j; |
David Turner | e49ab25 | 2000-05-16 23:44:38 +0000 | [diff] [blame] | 1043 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1044 | |
David Turner | c03d9cf | 2002-02-27 23:10:19 +0000 | [diff] [blame] | 1045 | FT_TRACE3(( "(%2d %2d %4x %2d) ", |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1046 | cur->platformID, |
| 1047 | cur->encodingID, |
| 1048 | cur->languageID, |
| 1049 | cur->nameID )); |
David Turner | e49ab25 | 2000-05-16 23:44:38 +0000 | [diff] [blame] | 1050 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1051 | /* I know that M$ encoded strings are Unicode, */ |
| 1052 | /* but this works reasonable well for debugging purposes. */ |
| 1053 | if ( cur->string ) |
David Turner | d3c8e06 | 2000-12-04 22:53:55 +0000 | [diff] [blame] | 1054 | for ( j = 0; j < (FT_UInt)cur->stringLength; j++ ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1055 | { |
David Turner | c03d9cf | 2002-02-27 23:10:19 +0000 | [diff] [blame] | 1056 | FT_Byte c = *(FT_Byte*)(cur->string + j); |
David Turner | e49ab25 | 2000-05-16 23:44:38 +0000 | [diff] [blame] | 1057 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1058 | |
David Turner | c03d9cf | 2002-02-27 23:10:19 +0000 | [diff] [blame] | 1059 | if ( c >= 32 && c < 128 ) |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1060 | FT_TRACE3(( "%c", c )); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1061 | } |
David Turner | c03d9cf | 2002-02-27 23:10:19 +0000 | [diff] [blame] | 1062 | else |
| 1063 | FT_TRACE3(( "INVALID ENTRY !!\n" )); |
| 1064 | |
| 1065 | FT_TRACE3(( "\n" )); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1066 | } |
| 1067 | } |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1068 | |
| 1069 | #endif /* FT_DEBUG_LEVEL_TRACE */ |
| 1070 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1071 | FT_TRACE2(( "loaded\n" )); |
| 1072 | |
David Turner | 49bd4f0 | 2000-07-12 16:57:37 +0000 | [diff] [blame] | 1073 | /* everything went well, update face->num_names */ |
| 1074 | face->num_names = names->numNameRecords; |
| 1075 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1076 | Exit: |
| 1077 | return error; |
| 1078 | } |
| 1079 | |
| 1080 | |
| 1081 | /*************************************************************************/ |
| 1082 | /* */ |
| 1083 | /* <Function> */ |
| 1084 | /* TT_Free_Names */ |
| 1085 | /* */ |
| 1086 | /* <Description> */ |
| 1087 | /* Frees the name records. */ |
| 1088 | /* */ |
| 1089 | /* <Input> */ |
| 1090 | /* face :: A handle to the target face object. */ |
| 1091 | /* */ |
David Turner | bc82f1b | 2002-03-01 02:26:22 +0000 | [diff] [blame] | 1092 | FT_LOCAL_DEF( void ) |
Werner Lemberg | 52005c3 | 2001-06-27 23:25:46 +0000 | [diff] [blame] | 1093 | TT_Free_Names( TT_Face face ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1094 | { |
David Turner | f0df85b | 2000-06-22 00:17:42 +0000 | [diff] [blame] | 1095 | FT_Memory memory = face->root.driver->root.memory; |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 1096 | TT_NameTable names = &face->name_table; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1097 | |
| 1098 | |
| 1099 | /* free strings table */ |
| 1100 | FREE( names->names ); |
| 1101 | |
| 1102 | /* free strings storage */ |
| 1103 | FREE( names->storage ); |
| 1104 | |
| 1105 | names->numNameRecords = 0; |
| 1106 | names->format = 0; |
| 1107 | names->storageOffset = 0; |
| 1108 | } |
| 1109 | |
| 1110 | |
| 1111 | /*************************************************************************/ |
| 1112 | /* */ |
| 1113 | /* <Function> */ |
| 1114 | /* TT_Load_CMap */ |
| 1115 | /* */ |
| 1116 | /* <Description> */ |
| 1117 | /* Loads the cmap directory in a face object. The cmaps itselves are */ |
| 1118 | /* loaded on demand in the `ttcmap.c' module. */ |
| 1119 | /* */ |
| 1120 | /* <Input> */ |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1121 | /* face :: A handle to the target face object. */ |
| 1122 | /* stream :: A handle to the input stream. */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1123 | /* */ |
| 1124 | /* <Return> */ |
Werner Lemberg | a929ba9 | 2000-06-25 06:47:11 +0000 | [diff] [blame] | 1125 | /* FreeType error code. 0 means success. */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1126 | /* */ |
David Turner | bc82f1b | 2002-03-01 02:26:22 +0000 | [diff] [blame] | 1127 | FT_LOCAL_DEF( FT_Error ) |
Werner Lemberg | 52005c3 | 2001-06-27 23:25:46 +0000 | [diff] [blame] | 1128 | TT_Load_CMap( TT_Face face, |
| 1129 | FT_Stream stream ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1130 | { |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 1131 | FT_Error error; |
| 1132 | FT_Memory memory = stream->memory; |
| 1133 | FT_Long table_start; |
| 1134 | TT_CMapDirRec cmap_dir; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1135 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1136 | const FT_Frame_Field cmap_fields[] = |
| 1137 | { |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 1138 | #undef FT_STRUCTURE |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 1139 | #define FT_STRUCTURE TT_CMapDirRec |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 1140 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1141 | FT_FRAME_START( 4 ), |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 1142 | FT_FRAME_USHORT( tableVersionNumber ), |
| 1143 | FT_FRAME_USHORT( numCMaps ), |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1144 | FT_FRAME_END |
| 1145 | }; |
David Turner | a56489e | 2000-02-13 13:41:56 +0000 | [diff] [blame] | 1146 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1147 | const FT_Frame_Field cmap_rec_fields[] = |
| 1148 | { |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 1149 | #undef FT_STRUCTURE |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 1150 | #define FT_STRUCTURE TT_CMapTableRec |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 1151 | |
Werner Lemberg | bcf8d4b | 2001-08-24 23:11:34 +0000 | [diff] [blame] | 1152 | FT_FRAME_START( 4 ), |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 1153 | FT_FRAME_USHORT( format ), |
| 1154 | FT_FRAME_USHORT( length ), |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1155 | FT_FRAME_END |
| 1156 | }; |
| 1157 | |
David Turner | a56489e | 2000-02-13 13:41:56 +0000 | [diff] [blame] | 1158 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1159 | FT_TRACE2(( "CMaps " )); |
| 1160 | |
| 1161 | error = face->goto_table( face, TTAG_cmap, stream, 0 ); |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1162 | if ( error ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1163 | { |
Werner Lemberg | 1f7f0e8 | 2001-06-06 17:30:41 +0000 | [diff] [blame] | 1164 | error = SFNT_Err_CMap_Table_Missing; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1165 | goto Exit; |
| 1166 | } |
| 1167 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1168 | table_start = FT_STREAM_POS(); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1169 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1170 | if ( FT_STREAM_READ_FIELDS( cmap_fields, &cmap_dir ) ) |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1171 | goto Exit; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1172 | |
Werner Lemberg | fbeb41d | 2000-07-02 00:27:53 +0000 | [diff] [blame] | 1173 | /* reserve space in face table for cmap tables */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1174 | if ( ALLOC_ARRAY( face->charmaps, |
| 1175 | cmap_dir.numCMaps, |
| 1176 | TT_CharMapRec ) ) |
| 1177 | goto Exit; |
| 1178 | |
| 1179 | face->num_charmaps = cmap_dir.numCMaps; |
| 1180 | { |
| 1181 | TT_CharMap charmap = face->charmaps; |
| 1182 | TT_CharMap limit = charmap + face->num_charmaps; |
| 1183 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1184 | |
David Turner | a56489e | 2000-02-13 13:41:56 +0000 | [diff] [blame] | 1185 | /* read the header of each charmap first */ |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1186 | if ( FT_FRAME_ENTER( face->num_charmaps * 8L ) ) |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1187 | goto Exit; |
| 1188 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1189 | for ( ; charmap < limit; charmap++ ) |
| 1190 | { |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 1191 | TT_CMapTable cmap; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1192 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1193 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1194 | charmap->root.face = (FT_Face)face; |
| 1195 | cmap = &charmap->cmap; |
| 1196 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1197 | cmap->loaded = FALSE; |
| 1198 | cmap->platformID = GET_UShort(); |
| 1199 | cmap->platformEncodingID = GET_UShort(); |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 1200 | cmap->offset = (FT_ULong)GET_Long(); |
David Turner | a56489e | 2000-02-13 13:41:56 +0000 | [diff] [blame] | 1201 | } |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1202 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1203 | FT_FRAME_EXIT(); |
David Turner | e49ab25 | 2000-05-16 23:44:38 +0000 | [diff] [blame] | 1204 | |
David Turner | a56489e | 2000-02-13 13:41:56 +0000 | [diff] [blame] | 1205 | /* now read the rest of each table */ |
| 1206 | for ( charmap = face->charmaps; charmap < limit; charmap++ ) |
| 1207 | { |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 1208 | TT_CMapTable cmap = &charmap->cmap; |
David Turner | e49ab25 | 2000-05-16 23:44:38 +0000 | [diff] [blame] | 1209 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1210 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1211 | if ( FT_STREAM_SEEK( table_start + (FT_Long)cmap->offset ) || |
| 1212 | FT_STREAM_READ_FIELDS( cmap_rec_fields, cmap ) ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1213 | goto Exit; |
David Turner | e49ab25 | 2000-05-16 23:44:38 +0000 | [diff] [blame] | 1214 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1215 | cmap->offset = FT_STREAM_POS(); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1216 | } |
| 1217 | } |
| 1218 | |
| 1219 | FT_TRACE2(( "loaded\n" )); |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1220 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1221 | Exit: |
| 1222 | return error; |
| 1223 | } |
| 1224 | |
| 1225 | |
| 1226 | /*************************************************************************/ |
| 1227 | /* */ |
| 1228 | /* <Function> */ |
| 1229 | /* TT_Load_OS2 */ |
| 1230 | /* */ |
| 1231 | /* <Description> */ |
| 1232 | /* Loads the OS2 table. */ |
| 1233 | /* */ |
| 1234 | /* <Input> */ |
| 1235 | /* face :: A handle to the target face object. */ |
| 1236 | /* stream :: A handle to the input stream. */ |
| 1237 | /* */ |
| 1238 | /* <Return> */ |
Werner Lemberg | a929ba9 | 2000-06-25 06:47:11 +0000 | [diff] [blame] | 1239 | /* FreeType error code. 0 means success. */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1240 | /* */ |
David Turner | bc82f1b | 2002-03-01 02:26:22 +0000 | [diff] [blame] | 1241 | FT_LOCAL_DEF( FT_Error ) |
Werner Lemberg | 52005c3 | 2001-06-27 23:25:46 +0000 | [diff] [blame] | 1242 | TT_Load_OS2( TT_Face face, |
| 1243 | FT_Stream stream ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1244 | { |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 1245 | FT_Error error; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1246 | TT_OS2* os2; |
David Turner | b1677a8 | 2000-05-29 20:37:41 +0000 | [diff] [blame] | 1247 | |
Werner Lemberg | 36af3ea | 2001-07-17 12:37:54 +0000 | [diff] [blame] | 1248 | const FT_Frame_Field os2_fields[] = |
| 1249 | { |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 1250 | #undef FT_STRUCTURE |
| 1251 | #define FT_STRUCTURE TT_OS2 |
| 1252 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1253 | FT_FRAME_START( 78 ), |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 1254 | FT_FRAME_USHORT( version ), |
| 1255 | FT_FRAME_SHORT ( xAvgCharWidth ), |
| 1256 | FT_FRAME_USHORT( usWeightClass ), |
| 1257 | FT_FRAME_USHORT( usWidthClass ), |
| 1258 | FT_FRAME_SHORT ( fsType ), |
| 1259 | FT_FRAME_SHORT ( ySubscriptXSize ), |
| 1260 | FT_FRAME_SHORT ( ySubscriptYSize ), |
| 1261 | FT_FRAME_SHORT ( ySubscriptXOffset ), |
| 1262 | FT_FRAME_SHORT ( ySubscriptYOffset ), |
| 1263 | FT_FRAME_SHORT ( ySuperscriptXSize ), |
| 1264 | FT_FRAME_SHORT ( ySuperscriptYSize ), |
| 1265 | FT_FRAME_SHORT ( ySuperscriptXOffset ), |
| 1266 | FT_FRAME_SHORT ( ySuperscriptYOffset ), |
| 1267 | FT_FRAME_SHORT ( yStrikeoutSize ), |
| 1268 | FT_FRAME_SHORT ( yStrikeoutPosition ), |
| 1269 | FT_FRAME_SHORT ( sFamilyClass ), |
| 1270 | FT_FRAME_BYTE ( panose[0] ), |
| 1271 | FT_FRAME_BYTE ( panose[1] ), |
| 1272 | FT_FRAME_BYTE ( panose[2] ), |
| 1273 | FT_FRAME_BYTE ( panose[3] ), |
| 1274 | FT_FRAME_BYTE ( panose[4] ), |
| 1275 | FT_FRAME_BYTE ( panose[5] ), |
| 1276 | FT_FRAME_BYTE ( panose[6] ), |
| 1277 | FT_FRAME_BYTE ( panose[7] ), |
| 1278 | FT_FRAME_BYTE ( panose[8] ), |
| 1279 | FT_FRAME_BYTE ( panose[9] ), |
| 1280 | FT_FRAME_ULONG ( ulUnicodeRange1 ), |
| 1281 | FT_FRAME_ULONG ( ulUnicodeRange2 ), |
| 1282 | FT_FRAME_ULONG ( ulUnicodeRange3 ), |
| 1283 | FT_FRAME_ULONG ( ulUnicodeRange4 ), |
| 1284 | FT_FRAME_BYTE ( achVendID[0] ), |
| 1285 | FT_FRAME_BYTE ( achVendID[1] ), |
| 1286 | FT_FRAME_BYTE ( achVendID[2] ), |
| 1287 | FT_FRAME_BYTE ( achVendID[3] ), |
David Turner | a56489e | 2000-02-13 13:41:56 +0000 | [diff] [blame] | 1288 | |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 1289 | FT_FRAME_USHORT( fsSelection ), |
| 1290 | FT_FRAME_USHORT( usFirstCharIndex ), |
| 1291 | FT_FRAME_USHORT( usLastCharIndex ), |
| 1292 | FT_FRAME_SHORT ( sTypoAscender ), |
| 1293 | FT_FRAME_SHORT ( sTypoDescender ), |
| 1294 | FT_FRAME_SHORT ( sTypoLineGap ), |
| 1295 | FT_FRAME_USHORT( usWinAscent ), |
| 1296 | FT_FRAME_USHORT( usWinDescent ), |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1297 | FT_FRAME_END |
| 1298 | }; |
David Turner | e49ab25 | 2000-05-16 23:44:38 +0000 | [diff] [blame] | 1299 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1300 | const FT_Frame_Field os2_fields_extra[] = |
| 1301 | { |
| 1302 | FT_FRAME_START( 8 ), |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 1303 | FT_FRAME_ULONG( ulCodePageRange1 ), |
| 1304 | FT_FRAME_ULONG( ulCodePageRange2 ), |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1305 | FT_FRAME_END |
| 1306 | }; |
David Turner | 1119bae | 2000-05-02 11:01:49 +0000 | [diff] [blame] | 1307 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1308 | const FT_Frame_Field os2_fields_extra2[] = |
| 1309 | { |
| 1310 | FT_FRAME_START( 10 ), |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 1311 | FT_FRAME_SHORT ( sxHeight ), |
| 1312 | FT_FRAME_SHORT ( sCapHeight ), |
| 1313 | FT_FRAME_USHORT( usDefaultChar ), |
| 1314 | FT_FRAME_USHORT( usBreakChar ), |
| 1315 | FT_FRAME_USHORT( usMaxContext ), |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1316 | FT_FRAME_END |
| 1317 | }; |
| 1318 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1319 | |
| 1320 | FT_TRACE2(( "OS/2 Table " )); |
| 1321 | |
| 1322 | /* We now support old Mac fonts where the OS/2 table doesn't */ |
| 1323 | /* exist. Simply put, we set the `version' field to 0xFFFF */ |
| 1324 | /* and test this value each time we need to access the table. */ |
| 1325 | error = face->goto_table( face, TTAG_OS2, stream, 0 ); |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1326 | if ( error ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1327 | { |
Werner Lemberg | fbeb41d | 2000-07-02 00:27:53 +0000 | [diff] [blame] | 1328 | FT_TRACE2(( "is missing!\n" )); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1329 | face->os2.version = 0xFFFF; |
Werner Lemberg | 1f7f0e8 | 2001-06-06 17:30:41 +0000 | [diff] [blame] | 1330 | error = SFNT_Err_Ok; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1331 | goto Exit; |
| 1332 | } |
| 1333 | |
David Turner | a56489e | 2000-02-13 13:41:56 +0000 | [diff] [blame] | 1334 | os2 = &face->os2; |
| 1335 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1336 | if ( FT_STREAM_READ_FIELDS( os2_fields, os2 ) ) |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1337 | goto Exit; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1338 | |
| 1339 | os2->ulCodePageRange1 = 0; |
| 1340 | os2->ulCodePageRange2 = 0; |
Werner Lemberg | 36af3ea | 2001-07-17 12:37:54 +0000 | [diff] [blame] | 1341 | os2->sxHeight = 0; |
| 1342 | os2->sCapHeight = 0; |
| 1343 | os2->usDefaultChar = 0; |
| 1344 | os2->usBreakChar = 0; |
| 1345 | os2->usMaxContext = 0; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1346 | |
| 1347 | if ( os2->version >= 0x0001 ) |
| 1348 | { |
| 1349 | /* only version 1 tables */ |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1350 | if ( FT_STREAM_READ_FIELDS( os2_fields_extra, os2 ) ) |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1351 | goto Exit; |
David Turner | 1119bae | 2000-05-02 11:01:49 +0000 | [diff] [blame] | 1352 | |
| 1353 | if ( os2->version >= 0x0002 ) |
| 1354 | { |
| 1355 | /* only version 2 tables */ |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1356 | if ( FT_STREAM_READ_FIELDS( os2_fields_extra2, os2 ) ) |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1357 | goto Exit; |
David Turner | 1119bae | 2000-05-02 11:01:49 +0000 | [diff] [blame] | 1358 | } |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1359 | } |
| 1360 | |
| 1361 | FT_TRACE2(( "loaded\n" )); |
| 1362 | |
| 1363 | Exit: |
| 1364 | return error; |
| 1365 | } |
| 1366 | |
| 1367 | |
| 1368 | /*************************************************************************/ |
| 1369 | /* */ |
| 1370 | /* <Function> */ |
| 1371 | /* TT_Load_Postscript */ |
| 1372 | /* */ |
| 1373 | /* <Description> */ |
| 1374 | /* Loads the Postscript table. */ |
| 1375 | /* */ |
| 1376 | /* <Input> */ |
| 1377 | /* face :: A handle to the target face object. */ |
| 1378 | /* stream :: A handle to the input stream. */ |
| 1379 | /* */ |
| 1380 | /* <Return> */ |
Werner Lemberg | a929ba9 | 2000-06-25 06:47:11 +0000 | [diff] [blame] | 1381 | /* FreeType error code. 0 means success. */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1382 | /* */ |
David Turner | bc82f1b | 2002-03-01 02:26:22 +0000 | [diff] [blame] | 1383 | FT_LOCAL_DEF( FT_Error ) |
Werner Lemberg | 52005c3 | 2001-06-27 23:25:46 +0000 | [diff] [blame] | 1384 | TT_Load_PostScript( TT_Face face, |
| 1385 | FT_Stream stream ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1386 | { |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 1387 | FT_Error error; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1388 | TT_Postscript* post = &face->postscript; |
David Turner | b1677a8 | 2000-05-29 20:37:41 +0000 | [diff] [blame] | 1389 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1390 | static const FT_Frame_Field post_fields[] = |
| 1391 | { |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 1392 | #undef FT_STRUCTURE |
| 1393 | #define FT_STRUCTURE TT_Postscript |
| 1394 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1395 | FT_FRAME_START( 32 ), |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 1396 | FT_FRAME_ULONG( FormatType ), |
| 1397 | FT_FRAME_ULONG( italicAngle ), |
| 1398 | FT_FRAME_SHORT( underlinePosition ), |
| 1399 | FT_FRAME_SHORT( underlineThickness ), |
| 1400 | FT_FRAME_ULONG( isFixedPitch ), |
| 1401 | FT_FRAME_ULONG( minMemType42 ), |
| 1402 | FT_FRAME_ULONG( maxMemType42 ), |
| 1403 | FT_FRAME_ULONG( minMemType1 ), |
| 1404 | FT_FRAME_ULONG( maxMemType1 ), |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1405 | FT_FRAME_END |
| 1406 | }; |
| 1407 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1408 | |
| 1409 | FT_TRACE2(( "PostScript " )); |
| 1410 | |
| 1411 | error = face->goto_table( face, TTAG_post, stream, 0 ); |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1412 | if ( error ) |
Werner Lemberg | 1f7f0e8 | 2001-06-06 17:30:41 +0000 | [diff] [blame] | 1413 | return SFNT_Err_Post_Table_Missing; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1414 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1415 | if ( FT_STREAM_READ_FIELDS( post_fields, post ) ) |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1416 | return error; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1417 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1418 | /* we don't load the glyph names, we do that in another */ |
| 1419 | /* module (ttpost). */ |
| 1420 | FT_TRACE2(( "loaded\n" )); |
| 1421 | |
Werner Lemberg | 1f7f0e8 | 2001-06-06 17:30:41 +0000 | [diff] [blame] | 1422 | return SFNT_Err_Ok; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1423 | } |
| 1424 | |
| 1425 | |
| 1426 | /*************************************************************************/ |
| 1427 | /* */ |
| 1428 | /* <Function> */ |
David Turner | 51179f0 | 2000-05-18 16:18:05 +0000 | [diff] [blame] | 1429 | /* TT_Load_PCLT */ |
| 1430 | /* */ |
| 1431 | /* <Description> */ |
| 1432 | /* Loads the PCL 5 Table. */ |
| 1433 | /* */ |
| 1434 | /* <Input> */ |
| 1435 | /* face :: A handle to the target face object. */ |
| 1436 | /* stream :: A handle to the input stream. */ |
| 1437 | /* */ |
| 1438 | /* <Return> */ |
Werner Lemberg | a929ba9 | 2000-06-25 06:47:11 +0000 | [diff] [blame] | 1439 | /* FreeType error code. 0 means success. */ |
David Turner | 51179f0 | 2000-05-18 16:18:05 +0000 | [diff] [blame] | 1440 | /* */ |
David Turner | bc82f1b | 2002-03-01 02:26:22 +0000 | [diff] [blame] | 1441 | FT_LOCAL_DEF( FT_Error ) |
Werner Lemberg | 52005c3 | 2001-06-27 23:25:46 +0000 | [diff] [blame] | 1442 | TT_Load_PCLT( TT_Face face, |
| 1443 | FT_Stream stream ) |
David Turner | 51179f0 | 2000-05-18 16:18:05 +0000 | [diff] [blame] | 1444 | { |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1445 | static const FT_Frame_Field pclt_fields[] = |
| 1446 | { |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 1447 | #undef FT_STRUCTURE |
| 1448 | #define FT_STRUCTURE TT_PCLT |
| 1449 | |
David Turner | d18388e | 2000-07-03 15:00:49 +0000 | [diff] [blame] | 1450 | FT_FRAME_START( 54 ), |
Werner Lemberg | e72c9fe | 2000-07-31 18:59:02 +0000 | [diff] [blame] | 1451 | FT_FRAME_ULONG ( Version ), |
| 1452 | FT_FRAME_ULONG ( FontNumber ), |
| 1453 | FT_FRAME_USHORT( Pitch ), |
| 1454 | FT_FRAME_USHORT( xHeight ), |
| 1455 | FT_FRAME_USHORT( Style ), |
| 1456 | FT_FRAME_USHORT( TypeFamily ), |
| 1457 | FT_FRAME_USHORT( CapHeight ), |
| 1458 | FT_FRAME_BYTES ( TypeFace, 16 ), |
| 1459 | FT_FRAME_BYTES ( CharacterComplement, 8 ), |
| 1460 | FT_FRAME_BYTES ( FileName, 6 ), |
| 1461 | FT_FRAME_CHAR ( StrokeWeight ), |
| 1462 | FT_FRAME_CHAR ( WidthType ), |
| 1463 | FT_FRAME_BYTE ( SerifStyle ), |
| 1464 | FT_FRAME_BYTE ( Reserved ), |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1465 | FT_FRAME_END |
| 1466 | }; |
Werner Lemberg | 920d41e | 2000-06-05 14:32:32 +0000 | [diff] [blame] | 1467 | |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 1468 | FT_Error error; |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1469 | TT_PCLT* pclt = &face->pclt; |
| 1470 | |
Werner Lemberg | 920d41e | 2000-06-05 14:32:32 +0000 | [diff] [blame] | 1471 | |
David Turner | 51179f0 | 2000-05-18 16:18:05 +0000 | [diff] [blame] | 1472 | FT_TRACE2(( "PCLT " )); |
| 1473 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1474 | /* optional table */ |
| 1475 | error = face->goto_table( face, TTAG_PCLT, stream, 0 ); |
| 1476 | if ( error ) |
| 1477 | { |
| 1478 | FT_TRACE2(( "missing (optional)\n" )); |
| 1479 | pclt->Version = 0; |
Werner Lemberg | 1f7f0e8 | 2001-06-06 17:30:41 +0000 | [diff] [blame] | 1480 | return SFNT_Err_Ok; |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1481 | } |
Werner Lemberg | 920d41e | 2000-06-05 14:32:32 +0000 | [diff] [blame] | 1482 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1483 | if ( FT_STREAM_READ_FIELDS( pclt_fields, pclt ) ) |
David Turner | 51179f0 | 2000-05-18 16:18:05 +0000 | [diff] [blame] | 1484 | goto Exit; |
Werner Lemberg | 920d41e | 2000-06-05 14:32:32 +0000 | [diff] [blame] | 1485 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1486 | FT_TRACE2(( "loaded\n" )); |
| 1487 | |
Werner Lemberg | 920d41e | 2000-06-05 14:32:32 +0000 | [diff] [blame] | 1488 | Exit: |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1489 | return error; |
| 1490 | } |
| 1491 | |
David Turner | 51179f0 | 2000-05-18 16:18:05 +0000 | [diff] [blame] | 1492 | |
| 1493 | /*************************************************************************/ |
| 1494 | /* */ |
| 1495 | /* <Function> */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1496 | /* TT_Load_Gasp */ |
| 1497 | /* */ |
| 1498 | /* <Description> */ |
Werner Lemberg | fbeb41d | 2000-07-02 00:27:53 +0000 | [diff] [blame] | 1499 | /* Loads the `gasp' table into a face object. */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1500 | /* */ |
| 1501 | /* <Input> */ |
| 1502 | /* face :: A handle to the target face object. */ |
| 1503 | /* stream :: The input stream. */ |
| 1504 | /* */ |
| 1505 | /* <Return> */ |
Werner Lemberg | a929ba9 | 2000-06-25 06:47:11 +0000 | [diff] [blame] | 1506 | /* FreeType error code. 0 means success. */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1507 | /* */ |
David Turner | bc82f1b | 2002-03-01 02:26:22 +0000 | [diff] [blame] | 1508 | FT_LOCAL_DEF( FT_Error ) |
Werner Lemberg | 52005c3 | 2001-06-27 23:25:46 +0000 | [diff] [blame] | 1509 | TT_Load_Gasp( TT_Face face, |
| 1510 | FT_Stream stream ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1511 | { |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 1512 | FT_Error error; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1513 | FT_Memory memory = stream->memory; |
| 1514 | |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 1515 | FT_UInt j,num_ranges; |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 1516 | TT_GaspRange gaspranges; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1517 | |
| 1518 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1519 | FT_TRACE2(( "TT_Load_Gasp: %08p\n", face )); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1520 | |
| 1521 | /* the gasp table is optional */ |
| 1522 | error = face->goto_table( face, TTAG_gasp, stream, 0 ); |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1523 | if ( error ) |
Werner Lemberg | 1f7f0e8 | 2001-06-06 17:30:41 +0000 | [diff] [blame] | 1524 | return SFNT_Err_Ok; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1525 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1526 | if ( FT_FRAME_ENTER( 4L ) ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1527 | goto Exit; |
| 1528 | |
| 1529 | face->gasp.version = GET_UShort(); |
| 1530 | face->gasp.numRanges = GET_UShort(); |
| 1531 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1532 | FT_FRAME_EXIT(); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1533 | |
| 1534 | num_ranges = face->gasp.numRanges; |
| 1535 | FT_TRACE3(( "number of ranges = %d\n", num_ranges )); |
| 1536 | |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 1537 | if ( ALLOC_ARRAY( gaspranges, num_ranges, TT_GaspRangeRec ) || |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1538 | FT_FRAME_ENTER( num_ranges * 4L ) ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1539 | goto Exit; |
| 1540 | |
| 1541 | face->gasp.gaspRanges = gaspranges; |
| 1542 | |
| 1543 | for ( j = 0; j < num_ranges; j++ ) |
| 1544 | { |
| 1545 | gaspranges[j].maxPPEM = GET_UShort(); |
| 1546 | gaspranges[j].gaspFlag = GET_UShort(); |
| 1547 | |
| 1548 | FT_TRACE3(( " [max:%d flag:%d]", |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1549 | gaspranges[j].maxPPEM, |
| 1550 | gaspranges[j].gaspFlag )); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1551 | } |
| 1552 | FT_TRACE3(( "\n" )); |
| 1553 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1554 | FT_FRAME_EXIT(); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1555 | FT_TRACE2(( "GASP loaded\n" )); |
| 1556 | |
| 1557 | Exit: |
| 1558 | return error; |
| 1559 | } |
| 1560 | |
| 1561 | |
Werner Lemberg | 52005c3 | 2001-06-27 23:25:46 +0000 | [diff] [blame] | 1562 | FT_CALLBACK_DEF( int ) |
| 1563 | tt_kern_pair_compare( const void* a, |
| 1564 | const void* b ); |
| 1565 | |
| 1566 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1567 | /*************************************************************************/ |
| 1568 | /* */ |
| 1569 | /* <Function> */ |
| 1570 | /* TT_Load_Kern */ |
| 1571 | /* */ |
| 1572 | /* <Description> */ |
| 1573 | /* Loads the first kerning table with format 0 in the font. Only */ |
| 1574 | /* accepts the first horizontal kerning table. Developers should use */ |
| 1575 | /* the `ftxkern' extension to access other kerning tables in the font */ |
| 1576 | /* file, if they really want to. */ |
| 1577 | /* */ |
| 1578 | /* <Input> */ |
| 1579 | /* face :: A handle to the target face object. */ |
| 1580 | /* stream :: The input stream. */ |
| 1581 | /* */ |
| 1582 | /* <Return> */ |
Werner Lemberg | a929ba9 | 2000-06-25 06:47:11 +0000 | [diff] [blame] | 1583 | /* FreeType error code. 0 means success. */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1584 | /* */ |
David Turner | bc82f1b | 2002-03-01 02:26:22 +0000 | [diff] [blame] | 1585 | FT_LOCAL_DEF( FT_Error ) |
Werner Lemberg | 52005c3 | 2001-06-27 23:25:46 +0000 | [diff] [blame] | 1586 | TT_Load_Kern( TT_Face face, |
| 1587 | FT_Stream stream ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1588 | { |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 1589 | FT_Error error; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1590 | FT_Memory memory = stream->memory; |
| 1591 | |
Werner Lemberg | cf24d51 | 2001-06-18 14:23:45 +0000 | [diff] [blame] | 1592 | FT_UInt n, num_tables; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1593 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1594 | |
Werner Lemberg | fbeb41d | 2000-07-02 00:27:53 +0000 | [diff] [blame] | 1595 | /* the kern table is optional; exit silently if it is missing */ |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1596 | error = face->goto_table( face, TTAG_kern, stream, 0 ); |
| 1597 | if ( error ) |
Werner Lemberg | 1f7f0e8 | 2001-06-06 17:30:41 +0000 | [diff] [blame] | 1598 | return SFNT_Err_Ok; |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1599 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1600 | if ( FT_FRAME_ENTER( 4L ) ) |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1601 | goto Exit; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1602 | |
Werner Lemberg | cf24d51 | 2001-06-18 14:23:45 +0000 | [diff] [blame] | 1603 | (void)GET_UShort(); /* version */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1604 | num_tables = GET_UShort(); |
| 1605 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1606 | FT_FRAME_EXIT(); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1607 | |
| 1608 | for ( n = 0; n < num_tables; n++ ) |
| 1609 | { |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 1610 | FT_UInt coverage; |
| 1611 | FT_UInt length; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1612 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1613 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1614 | if ( FT_FRAME_ENTER( 6L ) ) |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1615 | goto Exit; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1616 | |
Werner Lemberg | cf24d51 | 2001-06-18 14:23:45 +0000 | [diff] [blame] | 1617 | (void)GET_UShort(); /* version */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1618 | length = GET_UShort() - 6; /* substract header length */ |
| 1619 | coverage = GET_UShort(); |
| 1620 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1621 | FT_FRAME_EXIT(); |
David Turner | e49ab25 | 2000-05-16 23:44:38 +0000 | [diff] [blame] | 1622 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1623 | if ( coverage == 0x0001 ) |
| 1624 | { |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 1625 | FT_UInt num_pairs; |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 1626 | TT_Kern0_Pair pair; |
| 1627 | TT_Kern0_Pair limit; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1628 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1629 | |
| 1630 | /* found a horizontal format 0 kerning table! */ |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1631 | if ( FT_FRAME_ENTER( 8L ) ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1632 | goto Exit; |
| 1633 | |
| 1634 | num_pairs = GET_UShort(); |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1635 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1636 | /* skip the rest */ |
| 1637 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1638 | FT_FRAME_EXIT(); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1639 | |
| 1640 | /* allocate array of kerning pairs */ |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 1641 | if ( ALLOC_ARRAY( face->kern_pairs, num_pairs, TT_Kern0_PairRec ) || |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1642 | FT_FRAME_ENTER( 6L * num_pairs ) ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1643 | goto Exit; |
| 1644 | |
| 1645 | pair = face->kern_pairs; |
| 1646 | limit = pair + num_pairs; |
| 1647 | for ( ; pair < limit; pair++ ) |
| 1648 | { |
| 1649 | pair->left = GET_UShort(); |
| 1650 | pair->right = GET_UShort(); |
| 1651 | pair->value = GET_UShort(); |
| 1652 | } |
| 1653 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1654 | FT_FRAME_EXIT(); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1655 | |
| 1656 | face->num_kern_pairs = num_pairs; |
| 1657 | face->kern_table_index = n; |
Werner Lemberg | 415235d | 2001-06-28 17:49:10 +0000 | [diff] [blame] | 1658 | |
David Turner | 9a554eb | 2001-06-27 12:40:46 +0000 | [diff] [blame] | 1659 | /* ensure that the kerning pair table is sorted (yes, some */ |
Werner Lemberg | f814d0f | 2001-06-27 16:18:10 +0000 | [diff] [blame] | 1660 | /* fonts have unsorted tables!) */ |
David Turner | 9a554eb | 2001-06-27 12:40:46 +0000 | [diff] [blame] | 1661 | { |
| 1662 | FT_UInt i; |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 1663 | TT_Kern0_Pair pair0; |
Werner Lemberg | 415235d | 2001-06-28 17:49:10 +0000 | [diff] [blame] | 1664 | |
David Turner | 9a554eb | 2001-06-27 12:40:46 +0000 | [diff] [blame] | 1665 | |
Werner Lemberg | f814d0f | 2001-06-27 16:18:10 +0000 | [diff] [blame] | 1666 | pair0 = face->kern_pairs; |
| 1667 | |
| 1668 | for ( i = 1; i < num_pairs; i++, pair0++ ) |
David Turner | 9a554eb | 2001-06-27 12:40:46 +0000 | [diff] [blame] | 1669 | { |
Werner Lemberg | f814d0f | 2001-06-27 16:18:10 +0000 | [diff] [blame] | 1670 | if ( tt_kern_pair_compare( pair0, pair0 + 1 ) != -1 ) |
David Turner | 9a554eb | 2001-06-27 12:40:46 +0000 | [diff] [blame] | 1671 | { |
| 1672 | qsort( (void*)face->kern_pairs, (int)num_pairs, |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 1673 | sizeof ( TT_Kern0_PairRec ), tt_kern_pair_compare ); |
David Turner | 9a554eb | 2001-06-27 12:40:46 +0000 | [diff] [blame] | 1674 | break; |
| 1675 | } |
| 1676 | } |
| 1677 | } |
Werner Lemberg | 415235d | 2001-06-28 17:49:10 +0000 | [diff] [blame] | 1678 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1679 | goto Exit; |
| 1680 | } |
David Turner | e49ab25 | 2000-05-16 23:44:38 +0000 | [diff] [blame] | 1681 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1682 | if ( FT_STREAM_SKIP( length ) ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1683 | goto Exit; |
| 1684 | } |
| 1685 | |
| 1686 | /* no kern table found -- doesn't matter */ |
| 1687 | face->kern_table_index = -1; |
| 1688 | face->num_kern_pairs = 0; |
| 1689 | face->kern_pairs = NULL; |
| 1690 | |
| 1691 | Exit: |
| 1692 | return error; |
| 1693 | } |
| 1694 | |
Werner Lemberg | f814d0f | 2001-06-27 16:18:10 +0000 | [diff] [blame] | 1695 | |
David Turner | 9a554eb | 2001-06-27 12:40:46 +0000 | [diff] [blame] | 1696 | #undef TT_KERN_INDEX |
Werner Lemberg | f814d0f | 2001-06-27 16:18:10 +0000 | [diff] [blame] | 1697 | #define TT_KERN_INDEX( g1, g2 ) ( ( (FT_ULong)g1 << 16 ) | g2 ) |
David Turner | 9a554eb | 2001-06-27 12:40:46 +0000 | [diff] [blame] | 1698 | |
Werner Lemberg | 52005c3 | 2001-06-27 23:25:46 +0000 | [diff] [blame] | 1699 | FT_CALLBACK_DEF( int ) |
Werner Lemberg | f814d0f | 2001-06-27 16:18:10 +0000 | [diff] [blame] | 1700 | tt_kern_pair_compare( const void* a, |
| 1701 | const void* b ) |
David Turner | 9a554eb | 2001-06-27 12:40:46 +0000 | [diff] [blame] | 1702 | { |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 1703 | TT_Kern0_Pair pair1 = (TT_Kern0_Pair)a; |
| 1704 | TT_Kern0_Pair pair2 = (TT_Kern0_Pair)b; |
David Turner | 9a554eb | 2001-06-27 12:40:46 +0000 | [diff] [blame] | 1705 | |
| 1706 | FT_ULong index1 = TT_KERN_INDEX( pair1->left, pair1->right ); |
| 1707 | FT_ULong index2 = TT_KERN_INDEX( pair2->left, pair2->right ); |
| 1708 | |
Werner Lemberg | f814d0f | 2001-06-27 16:18:10 +0000 | [diff] [blame] | 1709 | |
David Turner | 9a554eb | 2001-06-27 12:40:46 +0000 | [diff] [blame] | 1710 | return ( index1 < index2 ? -1 : |
| 1711 | ( index1 > index2 ? 1 : 0 )); |
| 1712 | } |
| 1713 | |
| 1714 | #undef TT_KERN_INDEX |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1715 | |
Werner Lemberg | f814d0f | 2001-06-27 16:18:10 +0000 | [diff] [blame] | 1716 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1717 | /*************************************************************************/ |
| 1718 | /* */ |
| 1719 | /* <Function> */ |
| 1720 | /* TT_Load_Hdmx */ |
| 1721 | /* */ |
| 1722 | /* <Description> */ |
| 1723 | /* Loads the horizontal device metrics table. */ |
| 1724 | /* */ |
| 1725 | /* <Input> */ |
| 1726 | /* face :: A handle to the target face object. */ |
| 1727 | /* stream :: A handle to the input stream. */ |
| 1728 | /* */ |
| 1729 | /* <Return> */ |
Werner Lemberg | a929ba9 | 2000-06-25 06:47:11 +0000 | [diff] [blame] | 1730 | /* FreeType error code. 0 means success. */ |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1731 | /* */ |
David Turner | bc82f1b | 2002-03-01 02:26:22 +0000 | [diff] [blame] | 1732 | FT_LOCAL_DEF( FT_Error ) |
Werner Lemberg | 52005c3 | 2001-06-27 23:25:46 +0000 | [diff] [blame] | 1733 | TT_Load_Hdmx( TT_Face face, |
| 1734 | FT_Stream stream ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1735 | { |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 1736 | FT_Error error; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1737 | FT_Memory memory = stream->memory; |
| 1738 | |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 1739 | TT_Hdmx hdmx = &face->hdmx; |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 1740 | FT_Long num_glyphs; |
| 1741 | FT_Long record_size; |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1742 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1743 | |
| 1744 | hdmx->version = 0; |
| 1745 | hdmx->num_records = 0; |
| 1746 | hdmx->records = 0; |
| 1747 | |
| 1748 | /* this table is optional */ |
| 1749 | error = face->goto_table( face, TTAG_hdmx, stream, 0 ); |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1750 | if ( error ) |
Werner Lemberg | 1f7f0e8 | 2001-06-06 17:30:41 +0000 | [diff] [blame] | 1751 | return SFNT_Err_Ok; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1752 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1753 | if ( FT_FRAME_ENTER( 8L ) ) |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1754 | goto Exit; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1755 | |
| 1756 | hdmx->version = GET_UShort(); |
| 1757 | hdmx->num_records = GET_Short(); |
| 1758 | record_size = GET_Long(); |
| 1759 | |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1760 | FT_FRAME_EXIT(); |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1761 | |
| 1762 | /* Only recognize format 0 */ |
| 1763 | if ( hdmx->version != 0 ) |
| 1764 | goto Exit; |
| 1765 | |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 1766 | if ( ALLOC_ARRAY( hdmx->records, hdmx->num_records, TT_HdmxEntryRec ) ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1767 | goto Exit; |
| 1768 | |
| 1769 | num_glyphs = face->root.num_glyphs; |
| 1770 | record_size -= num_glyphs + 2; |
| 1771 | |
| 1772 | { |
David Turner | b5713c5 | 2002-03-14 11:26:29 +0000 | [diff] [blame] | 1773 | TT_HdmxEntry cur = hdmx->records; |
| 1774 | TT_HdmxEntry limit = cur + hdmx->num_records; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1775 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1776 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1777 | for ( ; cur < limit; cur++ ) |
| 1778 | { |
| 1779 | /* read record */ |
| 1780 | if ( READ_Byte( cur->ppem ) || |
| 1781 | READ_Byte( cur->max_width ) ) |
| 1782 | goto Exit; |
David Turner | e49ab25 | 2000-05-16 23:44:38 +0000 | [diff] [blame] | 1783 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1784 | if ( ALLOC( cur->widths, num_glyphs ) || |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1785 | FT_STREAM_READ( cur->widths, num_glyphs ) ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1786 | goto Exit; |
David Turner | e49ab25 | 2000-05-16 23:44:38 +0000 | [diff] [blame] | 1787 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1788 | /* skip padding bytes */ |
David Turner | 7d3a264 | 2002-03-20 10:49:31 +0000 | [diff] [blame^] | 1789 | if ( record_size > 0 && FT_STREAM_SKIP( record_size ) ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1790 | goto Exit; |
| 1791 | } |
| 1792 | } |
| 1793 | |
| 1794 | Exit: |
| 1795 | return error; |
| 1796 | } |
| 1797 | |
| 1798 | |
| 1799 | /*************************************************************************/ |
| 1800 | /* */ |
| 1801 | /* <Function> */ |
| 1802 | /* TT_Free_Hdmx */ |
| 1803 | /* */ |
| 1804 | /* <Description> */ |
| 1805 | /* Frees the horizontal device metrics table. */ |
| 1806 | /* */ |
| 1807 | /* <Input> */ |
| 1808 | /* face :: A handle to the target face object. */ |
| 1809 | /* */ |
David Turner | bc82f1b | 2002-03-01 02:26:22 +0000 | [diff] [blame] | 1810 | FT_LOCAL_DEF( void ) |
Werner Lemberg | 52005c3 | 2001-06-27 23:25:46 +0000 | [diff] [blame] | 1811 | TT_Free_Hdmx( TT_Face face ) |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1812 | { |
| 1813 | if ( face ) |
| 1814 | { |
David Turner | f9b8dec | 2000-06-16 19:34:52 +0000 | [diff] [blame] | 1815 | FT_Int n; |
David Turner | f0df85b | 2000-06-22 00:17:42 +0000 | [diff] [blame] | 1816 | FT_Memory memory = face->root.driver->root.memory; |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1817 | |
Werner Lemberg | 4e6dd85 | 2000-06-05 05:26:15 +0000 | [diff] [blame] | 1818 | |
David Turner | d2b1f35 | 1999-12-16 23:11:37 +0000 | [diff] [blame] | 1819 | for ( n = 0; n < face->hdmx.num_records; n++ ) |
| 1820 | FREE( face->hdmx.records[n].widths ); |
| 1821 | |
| 1822 | FREE( face->hdmx.records ); |
| 1823 | face->hdmx.num_records = 0; |
| 1824 | } |
| 1825 | } |
| 1826 | |
| 1827 | |
| 1828 | /* END */ |