1 /***************************************************************************/
5 /* PSNames module implementation (body). */
7 /* Copyright 1996-2001, 2002, 2003, 2005, 2006, 2007, 2008 by */
8 /* David Turner, Robert Wilhelm, and Werner Lemberg. */
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. */
16 /***************************************************************************/
20 #include FT_INTERNAL_OBJECTS_H
21 #include FT_SERVICE_POSTSCRIPT_CMAPS_H
30 #ifdef FT_CONFIG_OPTION_POSTSCRIPT_NAMES
33 #ifdef FT_CONFIG_OPTION_ADOBE_GLYPH_LIST
36 #define VARIANT_BIT 0x80000000UL
37 #define BASE_GLYPH( code ) ( (FT_UInt32)( (code) & ~VARIANT_BIT ) )
40 /* Return the Unicode value corresponding to a given glyph. Note that */
41 /* we do deal with glyph variants by detecting a non-initial dot in */
42 /* the name, as in `A.swash' or `e.final'; in this case, the */
43 /* VARIANT_BIT is set in the return value. */
46 ps_unicode_value( const char* glyph_name
)
48 /* If the name begins with `uni', then the glyph name may be a */
49 /* hard-coded unicode character code. */
50 if ( glyph_name
[0] == 'u' &&
51 glyph_name
[1] == 'n' &&
52 glyph_name
[2] == 'i' )
54 /* determine whether the next four characters following are */
57 /* XXX: Add code to deal with ligatures, i.e. glyph names like */
58 /* `uniXXXXYYYYZZZZ'... */
62 const char* p
= glyph_name
+ 3;
65 for ( count
= 4; count
> 0; count
--, p
++ )
71 d
= (unsigned char)c
- '0';
74 d
= (unsigned char)c
- 'A';
81 /* Exit if a non-uppercase hexadecimal character was found */
82 /* -- this also catches character codes below `0' since such */
83 /* negative numbers cast to `unsigned int' are far too big. */
87 value
= ( value
<< 4 ) + d
;
90 /* there must be exactly four hex digits */
96 return (FT_UInt32
)( value
| VARIANT_BIT
);
100 /* If the name begins with `u', followed by four to six uppercase */
101 /* hexadecimal digits, it is a hard-coded unicode character code. */
102 if ( glyph_name
[0] == 'u' )
106 const char* p
= glyph_name
+ 1;
109 for ( count
= 6; count
> 0; count
--, p
++ )
115 d
= (unsigned char)c
- '0';
118 d
= (unsigned char)c
- 'A';
128 value
= ( value
<< 4 ) + d
;
136 return (FT_UInt32
)( value
| VARIANT_BIT
);
140 /* Look for a non-initial dot in the glyph name in order to */
141 /* find variants like `A.swash', `e.final', etc. */
143 const char* p
= glyph_name
;
144 const char* dot
= NULL
;
149 if ( *p
== '.' && p
> glyph_name
)
156 /* now look up the glyph in the Adobe Glyph List */
158 return (FT_UInt32
)ft_get_adobe_glyph_index( glyph_name
, p
);
160 return (FT_UInt32
)( ft_get_adobe_glyph_index( glyph_name
, dot
) |
166 /* ft_qsort callback to sort the unicode map */
167 FT_CALLBACK_DEF( int )
168 compare_uni_maps( const void* a
,
171 PS_UniMap
* map1
= (PS_UniMap
*)a
;
172 PS_UniMap
* map2
= (PS_UniMap
*)b
;
173 FT_UInt32 unicode1
= BASE_GLYPH( map1
->unicode
);
174 FT_UInt32 unicode2
= BASE_GLYPH( map2
->unicode
);
177 /* sort base glyphs before glyph variants */
178 if ( unicode1
== unicode2
)
180 if ( map1
->unicode
> map2
->unicode
)
182 else if ( map1
->unicode
< map2
->unicode
)
189 if ( unicode1
> unicode2
)
191 else if ( unicode1
< unicode2
)
199 /* support for extra glyphs not handled (well) in AGL; */
200 /* we add extra mappings for them if necessary */
202 #define EXTRA_GLYPH_LIST_SIZE 10
204 static const FT_UInt32 ft_extra_glyph_unicodes
[EXTRA_GLYPH_LIST_SIZE
] =
220 static const char ft_extra_glyph_names
[] =
222 'D','e','l','t','a',0,
223 'O','m','e','g','a',0,
224 'f','r','a','c','t','i','o','n',0,
225 'h','y','p','h','e','n',0,
226 'm','a','c','r','o','n',0,
228 'p','e','r','i','o','d','c','e','n','t','e','r','e','d',0,
229 's','p','a','c','e',0,
230 'T','c','o','m','m','a','a','c','c','e','n','t',0,
231 't','c','o','m','m','a','a','c','c','e','n','t',0
235 ft_extra_glyph_name_offsets
[EXTRA_GLYPH_LIST_SIZE
] =
251 ps_check_extra_glyph_name( const char* gname
,
253 FT_UInt
* extra_glyphs
,
259 for ( n
= 0; n
< EXTRA_GLYPH_LIST_SIZE
; n
++ )
261 if ( ft_strcmp( ft_extra_glyph_names
+
262 ft_extra_glyph_name_offsets
[n
], gname
) == 0 )
264 if ( states
[n
] == 0 )
266 /* mark this extra glyph as a candidate for the cmap */
268 extra_glyphs
[n
] = glyph
;
278 ps_check_extra_glyph_unicode( FT_UInt32 uni_char
,
284 for ( n
= 0; n
< EXTRA_GLYPH_LIST_SIZE
; n
++ )
286 if ( uni_char
== ft_extra_glyph_unicodes
[n
] )
288 /* disable this extra glyph from being added to the cmap */
297 /* Build a table that maps Unicode values to glyph indices. */
299 ps_unicodes_init( FT_Memory memory
,
302 PS_GetGlyphNameFunc get_glyph_name
,
303 PS_FreeGlyphNameFunc free_glyph_name
,
304 FT_Pointer glyph_data
)
308 FT_UInt extra_glyph_list_states
[] = { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 };
309 FT_UInt extra_glyphs
[EXTRA_GLYPH_LIST_SIZE
];
312 /* we first allocate the table */
316 if ( !FT_NEW_ARRAY( table
->maps
, num_glyphs
+ EXTRA_GLYPH_LIST_SIZE
) )
326 for ( n
= 0; n
< num_glyphs
; n
++ )
328 const char* gname
= get_glyph_name( glyph_data
, n
);
333 ps_check_extra_glyph_name( gname
, n
,
334 extra_glyphs
, extra_glyph_list_states
);
335 uni_char
= ps_unicode_value( gname
);
337 if ( BASE_GLYPH( uni_char
) != 0 )
339 ps_check_extra_glyph_unicode( uni_char
,
340 extra_glyph_list_states
);
341 map
->unicode
= uni_char
;
342 map
->glyph_index
= n
;
346 if ( free_glyph_name
)
347 free_glyph_name( glyph_data
, gname
);
351 for ( n
= 0; n
< EXTRA_GLYPH_LIST_SIZE
; n
++ )
353 if ( extra_glyph_list_states
[n
] == 1 )
355 /* This glyph name has an additional representation. */
356 /* Add it to the cmap. */
358 map
->unicode
= ft_extra_glyph_unicodes
[n
];
359 map
->glyph_index
= extra_glyphs
[n
];
364 /* now compress the table a bit */
365 count
= (FT_UInt
)( map
- table
->maps
);
369 FT_FREE( table
->maps
);
371 error
= PSnames_Err_Invalid_Argument
; /* No unicode chars here! */
374 /* Reallocate if the number of used entries is much smaller. */
375 if ( count
< num_glyphs
/ 2 )
377 (void)FT_RENEW_ARRAY( table
->maps
, num_glyphs
, count
);
378 error
= PSnames_Err_Ok
;
381 /* Sort the table in increasing order of unicode values, */
382 /* taking care of glyph variants. */
383 ft_qsort( table
->maps
, count
, sizeof ( PS_UniMap
),
387 table
->num_maps
= count
;
395 ps_unicodes_char_index( PS_Unicodes table
,
398 PS_UniMap
*min
, *max
, *mid
, *result
= NULL
;
401 /* Perform a binary search on the table. */
404 max
= min
+ table
->num_maps
- 1;
408 FT_UInt32 base_glyph
;
411 mid
= min
+ ( ( max
- min
) >> 1 );
413 if ( mid
->unicode
== unicode
)
419 base_glyph
= BASE_GLYPH( mid
->unicode
);
421 if ( base_glyph
== unicode
)
422 result
= mid
; /* remember match but continue search for base glyph */
427 if ( base_glyph
< unicode
)
434 return result
->glyph_index
;
441 ps_unicodes_char_next( PS_Unicodes table
,
445 FT_UInt32 char_code
= *unicode
+ 1;
450 FT_UInt max
= table
->num_maps
;
453 FT_UInt32 base_glyph
;
458 mid
= min
+ ( ( max
- min
) >> 1 );
459 map
= table
->maps
+ mid
;
461 if ( map
->unicode
== char_code
)
463 result
= map
->glyph_index
;
467 base_glyph
= BASE_GLYPH( map
->unicode
);
469 if ( base_glyph
== char_code
)
470 result
= map
->glyph_index
;
472 if ( base_glyph
< char_code
)
479 goto Exit
; /* we have a variant glyph */
481 /* we didn't find it; check whether we have a map just above it */
484 if ( min
< table
->num_maps
)
486 map
= table
->maps
+ min
;
487 result
= map
->glyph_index
;
488 char_code
= BASE_GLYPH( map
->unicode
);
493 *unicode
= char_code
;
498 #endif /* FT_CONFIG_OPTION_ADOBE_GLYPH_LIST */
502 ps_get_macintosh_name( FT_UInt name_index
)
504 if ( name_index
>= FT_NUM_MAC_NAMES
)
507 return ft_standard_glyph_names
+ ft_mac_names
[name_index
];
512 ps_get_standard_strings( FT_UInt sid
)
514 if ( sid
>= FT_NUM_SID_NAMES
)
517 return ft_standard_glyph_names
+ ft_sid_names
[sid
];
521 #ifdef FT_CONFIG_OPTION_ADOBE_GLYPH_LIST
522 FT_DEFINE_SERVICE_PSCMAPSREC(pscmaps_interface
,
523 (PS_Unicode_ValueFunc
) ps_unicode_value
,
524 (PS_Unicodes_InitFunc
) ps_unicodes_init
,
525 (PS_Unicodes_CharIndexFunc
)ps_unicodes_char_index
,
526 (PS_Unicodes_CharNextFunc
) ps_unicodes_char_next
,
528 (PS_Macintosh_NameFunc
) ps_get_macintosh_name
,
529 (PS_Adobe_Std_StringsFunc
) ps_get_standard_strings
,
531 t1_standard_encoding
,
537 FT_DEFINE_SERVICE_PSCMAPSREC(pscmaps_interface
,
543 (PS_Macintosh_NameFunc
) ps_get_macintosh_name
,
544 (PS_Adobe_Std_StringsFunc
) ps_get_standard_strings
,
546 t1_standard_encoding
,
550 #endif /* FT_CONFIG_OPTION_ADOBE_GLYPH_LIST */
553 FT_DEFINE_SERVICEDESCREC1(pscmaps_services
,
554 FT_SERVICE_ID_POSTSCRIPT_CMAPS
, &FT_PSCMAPS_INTERFACE_GET
561 psnames_get_service( FT_Module module
,
562 const char* service_id
)
566 return ft_service_list_lookup( FT_PSCMAPS_SERVICES_GET
, service_id
);
569 #endif /* FT_CONFIG_OPTION_POSTSCRIPT_NAMES */
572 #ifndef FT_CONFIG_OPTION_POSTSCRIPT_NAMES
573 #define PUT_PS_NAMES_SERVICE(a) 0
575 #define PUT_PS_NAMES_SERVICE(a) a
578 FT_DEFINE_MODULE(psnames_module_class
,
580 0, /* this is not a font driver, nor a renderer */
581 sizeof ( FT_ModuleRec
),
583 "psnames", /* driver name */
584 0x10000L
, /* driver version */
585 0x20000L
, /* driver requires FreeType 2 or above */
587 PUT_PS_NAMES_SERVICE((void*)&FT_PSCMAPS_INTERFACE_GET
), /* module specific interface */
588 (FT_Module_Constructor
)0,
589 (FT_Module_Destructor
) 0,
590 (FT_Module_Requester
) PUT_PS_NAMES_SERVICE(psnames_get_service
)