1/****************************************************************************
2 *
3 * t1cmap.c
4 *
5 * Type 1 character map support (body).
6 *
7 * Copyright (C) 2002-2019 by
8 * David Turner, Robert Wilhelm, and Werner Lemberg.
9 *
10 * This file is part of the FreeType project, and may only be used,
11 * modified, and distributed under the terms of the FreeType project
12 * license, LICENSE.TXT. By continuing to use, modify, or distribute
13 * this file you indicate that you have read the license and
14 * understand and accept it fully.
15 *
16 */
17
18
19#include "t1cmap.h"
20
21#include FT_INTERNAL_DEBUG_H
22
23#include "psauxerr.h"
24
25
26 /*************************************************************************/
27 /*************************************************************************/
28 /***** *****/
29 /***** TYPE1 STANDARD (AND EXPERT) ENCODING CMAPS *****/
30 /***** *****/
31 /*************************************************************************/
32 /*************************************************************************/
33
34 static void
35 t1_cmap_std_init( T1_CMapStd cmap,
36 FT_Int is_expert )
37 {
38 T1_Face face = (T1_Face)FT_CMAP_FACE( cmap );
39 FT_Service_PsCMaps psnames = (FT_Service_PsCMaps)face->psnames;
40
41
42 cmap->num_glyphs = (FT_UInt)face->type1.num_glyphs;
43 cmap->glyph_names = (const char* const*)face->type1.glyph_names;
44 cmap->sid_to_string = psnames->adobe_std_strings;
45 cmap->code_to_sid = is_expert ? psnames->adobe_expert_encoding
46 : psnames->adobe_std_encoding;
47
48 FT_ASSERT( cmap->code_to_sid );
49 }
50
51
52 FT_CALLBACK_DEF( void )
53 t1_cmap_std_done( T1_CMapStd cmap )
54 {
55 cmap->num_glyphs = 0;
56 cmap->glyph_names = NULL;
57 cmap->sid_to_string = NULL;
58 cmap->code_to_sid = NULL;
59 }
60
61
62 FT_CALLBACK_DEF( FT_UInt )
63 t1_cmap_std_char_index( T1_CMapStd cmap,
64 FT_UInt32 char_code )
65 {
66 FT_UInt result = 0;
67
68
69 if ( char_code < 256 )
70 {
71 FT_UInt code, n;
72 const char* glyph_name;
73
74
75 /* convert character code to Adobe SID string */
76 code = cmap->code_to_sid[char_code];
77 glyph_name = cmap->sid_to_string( code );
78
79 /* look for the corresponding glyph name */
80 for ( n = 0; n < cmap->num_glyphs; n++ )
81 {
82 const char* gname = cmap->glyph_names[n];
83
84
85 if ( gname && gname[0] == glyph_name[0] &&
86 ft_strcmp( gname, glyph_name ) == 0 )
87 {
88 result = n;
89 break;
90 }
91 }
92 }
93
94 return result;
95 }
96
97
98 FT_CALLBACK_DEF( FT_UInt32 )
99 t1_cmap_std_char_next( T1_CMapStd cmap,
100 FT_UInt32 *pchar_code )
101 {
102 FT_UInt result = 0;
103 FT_UInt32 char_code = *pchar_code + 1;
104
105
106 while ( char_code < 256 )
107 {
108 result = t1_cmap_std_char_index( cmap, char_code );
109 if ( result != 0 )
110 goto Exit;
111
112 char_code++;
113 }
114 char_code = 0;
115
116 Exit:
117 *pchar_code = char_code;
118 return result;
119 }
120
121
122 FT_CALLBACK_DEF( FT_Error )
123 t1_cmap_standard_init( T1_CMapStd cmap,
124 FT_Pointer pointer )
125 {
126 FT_UNUSED( pointer );
127
128
129 t1_cmap_std_init( cmap, 0 );
130 return 0;
131 }
132
133
134 FT_CALLBACK_TABLE_DEF const FT_CMap_ClassRec
135 t1_cmap_standard_class_rec =
136 {
137 sizeof ( T1_CMapStdRec ),
138
139 (FT_CMap_InitFunc) t1_cmap_standard_init, /* init */
140 (FT_CMap_DoneFunc) t1_cmap_std_done, /* done */
141 (FT_CMap_CharIndexFunc)t1_cmap_std_char_index, /* char_index */
142 (FT_CMap_CharNextFunc) t1_cmap_std_char_next, /* char_next */
143
144 (FT_CMap_CharVarIndexFunc) NULL, /* char_var_index */
145 (FT_CMap_CharVarIsDefaultFunc)NULL, /* char_var_default */
146 (FT_CMap_VariantListFunc) NULL, /* variant_list */
147 (FT_CMap_CharVariantListFunc) NULL, /* charvariant_list */
148 (FT_CMap_VariantCharListFunc) NULL /* variantchar_list */
149 };
150
151
152 FT_CALLBACK_DEF( FT_Error )
153 t1_cmap_expert_init( T1_CMapStd cmap,
154 FT_Pointer pointer )
155 {
156 FT_UNUSED( pointer );
157
158
159 t1_cmap_std_init( cmap, 1 );
160 return 0;
161 }
162
163 FT_CALLBACK_TABLE_DEF const FT_CMap_ClassRec
164 t1_cmap_expert_class_rec =
165 {
166 sizeof ( T1_CMapStdRec ),
167
168 (FT_CMap_InitFunc) t1_cmap_expert_init, /* init */
169 (FT_CMap_DoneFunc) t1_cmap_std_done, /* done */
170 (FT_CMap_CharIndexFunc)t1_cmap_std_char_index, /* char_index */
171 (FT_CMap_CharNextFunc) t1_cmap_std_char_next, /* char_next */
172
173 (FT_CMap_CharVarIndexFunc) NULL, /* char_var_index */
174 (FT_CMap_CharVarIsDefaultFunc)NULL, /* char_var_default */
175 (FT_CMap_VariantListFunc) NULL, /* variant_list */
176 (FT_CMap_CharVariantListFunc) NULL, /* charvariant_list */
177 (FT_CMap_VariantCharListFunc) NULL /* variantchar_list */
178 };
179
180
181 /*************************************************************************/
182 /*************************************************************************/
183 /***** *****/
184 /***** TYPE1 CUSTOM ENCODING CMAP *****/
185 /***** *****/
186 /*************************************************************************/
187 /*************************************************************************/
188
189
190 FT_CALLBACK_DEF( FT_Error )
191 t1_cmap_custom_init( T1_CMapCustom cmap,
192 FT_Pointer pointer )
193 {
194 T1_Face face = (T1_Face)FT_CMAP_FACE( cmap );
195 T1_Encoding encoding = &face->type1.encoding;
196
197 FT_UNUSED( pointer );
198
199
200 cmap->first = (FT_UInt)encoding->code_first;
201 cmap->count = (FT_UInt)encoding->code_last - cmap->first;
202 cmap->indices = encoding->char_index;
203
204 FT_ASSERT( cmap->indices );
205 FT_ASSERT( encoding->code_first <= encoding->code_last );
206
207 return 0;
208 }
209
210
211 FT_CALLBACK_DEF( void )
212 t1_cmap_custom_done( T1_CMapCustom cmap )
213 {
214 cmap->indices = NULL;
215 cmap->first = 0;
216 cmap->count = 0;
217 }
218
219
220 FT_CALLBACK_DEF( FT_UInt )
221 t1_cmap_custom_char_index( T1_CMapCustom cmap,
222 FT_UInt32 char_code )
223 {
224 FT_UInt result = 0;
225
226
227 if ( ( char_code >= cmap->first ) &&
228 ( char_code < ( cmap->first + cmap->count ) ) )
229 result = cmap->indices[char_code];
230
231 return result;
232 }
233
234
235 FT_CALLBACK_DEF( FT_UInt32 )
236 t1_cmap_custom_char_next( T1_CMapCustom cmap,
237 FT_UInt32 *pchar_code )
238 {
239 FT_UInt result = 0;
240 FT_UInt32 char_code = *pchar_code;
241
242
243 char_code++;
244
245 if ( char_code < cmap->first )
246 char_code = cmap->first;
247
248 for ( ; char_code < ( cmap->first + cmap->count ); char_code++ )
249 {
250 result = cmap->indices[char_code];
251 if ( result != 0 )
252 goto Exit;
253 }
254
255 char_code = 0;
256
257 Exit:
258 *pchar_code = char_code;
259 return result;
260 }
261
262
263 FT_CALLBACK_TABLE_DEF const FT_CMap_ClassRec
264 t1_cmap_custom_class_rec =
265 {
266 sizeof ( T1_CMapCustomRec ),
267
268 (FT_CMap_InitFunc) t1_cmap_custom_init, /* init */
269 (FT_CMap_DoneFunc) t1_cmap_custom_done, /* done */
270 (FT_CMap_CharIndexFunc)t1_cmap_custom_char_index, /* char_index */
271 (FT_CMap_CharNextFunc) t1_cmap_custom_char_next, /* char_next */
272
273 (FT_CMap_CharVarIndexFunc) NULL, /* char_var_index */
274 (FT_CMap_CharVarIsDefaultFunc)NULL, /* char_var_default */
275 (FT_CMap_VariantListFunc) NULL, /* variant_list */
276 (FT_CMap_CharVariantListFunc) NULL, /* charvariant_list */
277 (FT_CMap_VariantCharListFunc) NULL /* variantchar_list */
278 };
279
280
281 /*************************************************************************/
282 /*************************************************************************/
283 /***** *****/
284 /***** TYPE1 SYNTHETIC UNICODE ENCODING CMAP *****/
285 /***** *****/
286 /*************************************************************************/
287 /*************************************************************************/
288
289 FT_CALLBACK_DEF( const char * )
290 psaux_get_glyph_name( T1_Face face,
291 FT_UInt idx )
292 {
293 return face->type1.glyph_names[idx];
294 }
295
296
297 FT_CALLBACK_DEF( FT_Error )
298 t1_cmap_unicode_init( PS_Unicodes unicodes,
299 FT_Pointer pointer )
300 {
301 T1_Face face = (T1_Face)FT_CMAP_FACE( unicodes );
302 FT_Memory memory = FT_FACE_MEMORY( face );
303 FT_Service_PsCMaps psnames = (FT_Service_PsCMaps)face->psnames;
304
305 FT_UNUSED( pointer );
306
307
308 if ( !psnames->unicodes_init )
309 return FT_THROW( Unimplemented_Feature );
310
311 return psnames->unicodes_init( memory,
312 unicodes,
313 (FT_UInt)face->type1.num_glyphs,
314 (PS_GetGlyphNameFunc)&psaux_get_glyph_name,
315 (PS_FreeGlyphNameFunc)NULL,
316 (FT_Pointer)face );
317 }
318
319
320 FT_CALLBACK_DEF( void )
321 t1_cmap_unicode_done( PS_Unicodes unicodes )
322 {
323 FT_Face face = FT_CMAP_FACE( unicodes );
324 FT_Memory memory = FT_FACE_MEMORY( face );
325
326
327 FT_FREE( unicodes->maps );
328 unicodes->num_maps = 0;
329 }
330
331
332 FT_CALLBACK_DEF( FT_UInt )
333 t1_cmap_unicode_char_index( PS_Unicodes unicodes,
334 FT_UInt32 char_code )
335 {
336 T1_Face face = (T1_Face)FT_CMAP_FACE( unicodes );
337 FT_Service_PsCMaps psnames = (FT_Service_PsCMaps)face->psnames;
338
339
340 return psnames->unicodes_char_index( unicodes, char_code );
341 }
342
343
344 FT_CALLBACK_DEF( FT_UInt32 )
345 t1_cmap_unicode_char_next( PS_Unicodes unicodes,
346 FT_UInt32 *pchar_code )
347 {
348 T1_Face face = (T1_Face)FT_CMAP_FACE( unicodes );
349 FT_Service_PsCMaps psnames = (FT_Service_PsCMaps)face->psnames;
350
351
352 return psnames->unicodes_char_next( unicodes, pchar_code );
353 }
354
355
356 FT_CALLBACK_TABLE_DEF const FT_CMap_ClassRec
357 t1_cmap_unicode_class_rec =
358 {
359 sizeof ( PS_UnicodesRec ),
360
361 (FT_CMap_InitFunc) t1_cmap_unicode_init, /* init */
362 (FT_CMap_DoneFunc) t1_cmap_unicode_done, /* done */
363 (FT_CMap_CharIndexFunc)t1_cmap_unicode_char_index, /* char_index */
364 (FT_CMap_CharNextFunc) t1_cmap_unicode_char_next, /* char_next */
365
366 (FT_CMap_CharVarIndexFunc) NULL, /* char_var_index */
367 (FT_CMap_CharVarIsDefaultFunc)NULL, /* char_var_default */
368 (FT_CMap_VariantListFunc) NULL, /* variant_list */
369 (FT_CMap_CharVariantListFunc) NULL, /* charvariant_list */
370 (FT_CMap_VariantCharListFunc) NULL /* variantchar_list */
371 };
372
373
374/* END */
375