• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /****************************************************************************
2  *
3  * ftccmap.c
4  *
5  *   FreeType CharMap cache (body)
6  *
7  * Copyright (C) 2000-2022 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 <freetype/freetype.h>
20 #include <freetype/ftcache.h>
21 #include "ftcmanag.h"
22 #include <freetype/internal/ftmemory.h>
23 #include <freetype/internal/ftobjs.h>
24 #include <freetype/internal/ftdebug.h>
25 
26 #include "ftccback.h"
27 #include "ftcerror.h"
28 
29 #undef  FT_COMPONENT
30 #define FT_COMPONENT  cache
31 
32 
33   /**************************************************************************
34    *
35    * Each FTC_CMapNode contains a simple array to map a range of character
36    * codes to equivalent glyph indices.
37    *
38    * For now, the implementation is very basic: Each node maps a range of
39    * 128 consecutive character codes to their corresponding glyph indices.
40    *
41    * We could do more complex things, but I don't think it is really very
42    * useful.
43    *
44    */
45 
46 
47   /* number of glyph indices / character code per node */
48 #define FTC_CMAP_INDICES_MAX  128
49 
50   /* compute a query/node hash */
51 #define FTC_CMAP_HASH( faceid, index, charcode )         \
52           ( FTC_FACE_ID_HASH( faceid ) + 211 * (index) + \
53             ( (charcode) / FTC_CMAP_INDICES_MAX )      )
54 
55   /* the charmap query */
56   typedef struct  FTC_CMapQueryRec_
57   {
58     FTC_FaceID  face_id;
59     FT_UInt     cmap_index;
60     FT_UInt32   char_code;
61 
62   } FTC_CMapQueryRec, *FTC_CMapQuery;
63 
64 #define FTC_CMAP_QUERY( x )  ((FTC_CMapQuery)(x))
65 
66   /* the cmap cache node */
67   typedef struct  FTC_CMapNodeRec_
68   {
69     FTC_NodeRec  node;
70     FTC_FaceID   face_id;
71     FT_UInt      cmap_index;
72     FT_UInt32    first;                         /* first character in node */
73     FT_UInt16    indices[FTC_CMAP_INDICES_MAX]; /* array of glyph indices  */
74 
75   } FTC_CMapNodeRec, *FTC_CMapNode;
76 
77 #define FTC_CMAP_NODE( x ) ( (FTC_CMapNode)( x ) )
78 
79   /* if (indices[n] == FTC_CMAP_UNKNOWN), we assume that the corresponding */
80   /* glyph indices haven't been queried through FT_Get_Glyph_Index() yet   */
81 #define FTC_CMAP_UNKNOWN  (FT_UInt16)~0
82 
83 
84   /*************************************************************************/
85   /*************************************************************************/
86   /*****                                                               *****/
87   /*****                        CHARMAP NODES                          *****/
88   /*****                                                               *****/
89   /*************************************************************************/
90   /*************************************************************************/
91 
92 
93   FT_CALLBACK_DEF( void )
ftc_cmap_node_free(FTC_Node ftcnode,FTC_Cache cache)94   ftc_cmap_node_free( FTC_Node   ftcnode,
95                       FTC_Cache  cache )
96   {
97     FTC_CMapNode  node   = (FTC_CMapNode)ftcnode;
98     FT_Memory     memory = cache->memory;
99 
100 
101     FT_FREE( node );
102   }
103 
104 
105   /* initialize a new cmap node */
106   FT_CALLBACK_DEF( FT_Error )
ftc_cmap_node_new(FTC_Node * ftcanode,FT_Pointer ftcquery,FTC_Cache cache)107   ftc_cmap_node_new( FTC_Node   *ftcanode,
108                      FT_Pointer  ftcquery,
109                      FTC_Cache   cache )
110   {
111     FTC_CMapNode  *anode  = (FTC_CMapNode*)ftcanode;
112     FTC_CMapQuery  query  = (FTC_CMapQuery)ftcquery;
113     FT_Error       error;
114     FT_Memory      memory = cache->memory;
115     FTC_CMapNode   node   = NULL;
116     FT_UInt        nn;
117 
118 
119     if ( !FT_QNEW( node ) )
120     {
121       node->face_id    = query->face_id;
122       node->cmap_index = query->cmap_index;
123       node->first      = (query->char_code / FTC_CMAP_INDICES_MAX) *
124                          FTC_CMAP_INDICES_MAX;
125 
126       for ( nn = 0; nn < FTC_CMAP_INDICES_MAX; nn++ )
127         node->indices[nn] = FTC_CMAP_UNKNOWN;
128     }
129 
130     *anode = node;
131     return error;
132   }
133 
134 
135   /* compute the weight of a given cmap node */
136   FT_CALLBACK_DEF( FT_Offset )
ftc_cmap_node_weight(FTC_Node cnode,FTC_Cache cache)137   ftc_cmap_node_weight( FTC_Node   cnode,
138                         FTC_Cache  cache )
139   {
140     FT_UNUSED( cnode );
141     FT_UNUSED( cache );
142 
143     return sizeof ( *cnode );
144   }
145 
146 
147   /* compare a cmap node to a given query */
148   FT_CALLBACK_DEF( FT_Bool )
ftc_cmap_node_compare(FTC_Node ftcnode,FT_Pointer ftcquery,FTC_Cache cache,FT_Bool * list_changed)149   ftc_cmap_node_compare( FTC_Node    ftcnode,
150                          FT_Pointer  ftcquery,
151                          FTC_Cache   cache,
152                          FT_Bool*    list_changed )
153   {
154     FTC_CMapNode   node  = (FTC_CMapNode)ftcnode;
155     FTC_CMapQuery  query = (FTC_CMapQuery)ftcquery;
156     FT_UNUSED( cache );
157 
158 
159     if ( list_changed )
160       *list_changed = FALSE;
161     if ( node->face_id    == query->face_id    &&
162          node->cmap_index == query->cmap_index )
163     {
164       FT_UInt32  offset = (FT_UInt32)( query->char_code - node->first );
165 
166 
167       return FT_BOOL( offset < FTC_CMAP_INDICES_MAX );
168     }
169 
170     return 0;
171   }
172 
173 
174   FT_CALLBACK_DEF( FT_Bool )
ftc_cmap_node_remove_faceid(FTC_Node ftcnode,FT_Pointer ftcface_id,FTC_Cache cache,FT_Bool * list_changed)175   ftc_cmap_node_remove_faceid( FTC_Node    ftcnode,
176                                FT_Pointer  ftcface_id,
177                                FTC_Cache   cache,
178                                FT_Bool*    list_changed )
179   {
180     FTC_CMapNode  node    = (FTC_CMapNode)ftcnode;
181     FTC_FaceID    face_id = (FTC_FaceID)ftcface_id;
182     FT_UNUSED( cache );
183 
184 
185     if ( list_changed )
186       *list_changed = FALSE;
187     return FT_BOOL( node->face_id == face_id );
188   }
189 
190 
191   /*************************************************************************/
192   /*************************************************************************/
193   /*****                                                               *****/
194   /*****                    GLYPH IMAGE CACHE                          *****/
195   /*****                                                               *****/
196   /*************************************************************************/
197   /*************************************************************************/
198 
199 
200   static
201   const FTC_CacheClassRec  ftc_cmap_cache_class =
202   {
203     ftc_cmap_node_new,           /* FTC_Node_NewFunc      node_new           */
204     ftc_cmap_node_weight,        /* FTC_Node_WeightFunc   node_weight        */
205     ftc_cmap_node_compare,       /* FTC_Node_CompareFunc  node_compare       */
206     ftc_cmap_node_remove_faceid, /* FTC_Node_CompareFunc  node_remove_faceid */
207     ftc_cmap_node_free,          /* FTC_Node_FreeFunc     node_free          */
208 
209     sizeof ( FTC_CacheRec ),
210     ftc_cache_init,              /* FTC_Cache_InitFunc    cache_init         */
211     ftc_cache_done,              /* FTC_Cache_DoneFunc    cache_done         */
212   };
213 
214 
215   /* documentation is in ftcache.h */
216 
217   FT_EXPORT_DEF( FT_Error )
FTC_CMapCache_New(FTC_Manager manager,FTC_CMapCache * acache)218   FTC_CMapCache_New( FTC_Manager     manager,
219                      FTC_CMapCache  *acache )
220   {
221     return FTC_Manager_RegisterCache( manager,
222                                       &ftc_cmap_cache_class,
223                                       FTC_CACHE_P( acache ) );
224   }
225 
226 
227   /* documentation is in ftcache.h */
228 
229   FT_EXPORT_DEF( FT_UInt )
FTC_CMapCache_Lookup(FTC_CMapCache cmap_cache,FTC_FaceID face_id,FT_Int cmap_index,FT_UInt32 char_code)230   FTC_CMapCache_Lookup( FTC_CMapCache  cmap_cache,
231                         FTC_FaceID     face_id,
232                         FT_Int         cmap_index,
233                         FT_UInt32      char_code )
234   {
235     FTC_Cache         cache = FTC_CACHE( cmap_cache );
236     FTC_CMapQueryRec  query;
237     FTC_Node          node;
238     FT_Error          error;
239     FT_UInt           gindex = 0;
240     FT_Offset         hash;
241     FT_Int            no_cmap_change = 0;
242 
243 
244     if ( cmap_index < 0 )
245     {
246       /* Treat a negative cmap index as a special value, meaning that you */
247       /* don't want to change the FT_Face's character map through this    */
248       /* call.  This can be useful if the face requester callback already */
249       /* sets the face's charmap to the appropriate value.                */
250 
251       no_cmap_change = 1;
252       cmap_index     = 0;
253     }
254 
255     if ( !cache )
256     {
257       FT_TRACE0(( "FTC_CMapCache_Lookup: bad arguments, returning 0\n" ));
258       return 0;
259     }
260 
261     query.face_id    = face_id;
262     query.cmap_index = (FT_UInt)cmap_index;
263     query.char_code  = char_code;
264 
265     hash = FTC_CMAP_HASH( face_id, (FT_UInt)cmap_index, char_code );
266 
267 #if 1
268     FTC_CACHE_LOOKUP_CMP( cache, ftc_cmap_node_compare, hash, &query,
269                           node, error );
270 #else
271     error = FTC_Cache_Lookup( cache, hash, &query, &node );
272 #endif
273     if ( error )
274       goto Exit;
275 
276     FT_ASSERT( char_code - FTC_CMAP_NODE( node )->first <
277                FTC_CMAP_INDICES_MAX );
278 
279     /* something rotten can happen with rogue clients */
280     if ( char_code - FTC_CMAP_NODE( node )->first >= FTC_CMAP_INDICES_MAX )
281       return 0; /* XXX: should return appropriate error */
282 
283     gindex = FTC_CMAP_NODE( node )->indices[char_code -
284                                             FTC_CMAP_NODE( node )->first];
285     if ( gindex == FTC_CMAP_UNKNOWN )
286     {
287       FT_Face  face;
288 
289 
290       gindex = 0;
291 
292       error = FTC_Manager_LookupFace( cache->manager,
293                                       FTC_CMAP_NODE( node )->face_id,
294                                       &face );
295       if ( error )
296         goto Exit;
297 
298       if ( (FT_UInt)cmap_index < (FT_UInt)face->num_charmaps )
299       {
300         FT_CharMap  old, cmap  = NULL;
301 
302 
303         old  = face->charmap;
304         cmap = face->charmaps[cmap_index];
305 
306         if ( old != cmap && !no_cmap_change )
307           FT_Set_Charmap( face, cmap );
308 
309         gindex = FT_Get_Char_Index( face, char_code );
310 
311         if ( old != cmap && !no_cmap_change )
312           FT_Set_Charmap( face, old );
313       }
314 
315       FTC_CMAP_NODE( node )->indices[char_code -
316                                      FTC_CMAP_NODE( node )->first]
317         = (FT_UShort)gindex;
318     }
319 
320   Exit:
321     return gindex;
322   }
323 
324 
325 /* END */
326