1 /***************************************************************************/
5 /* The FreeType services (specification only). */
7 /* Copyright 2003-2007, 2009, 2012 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 /***************************************************************************/
18 /*************************************************************************/
20 /* Each module can export one or more `services'. Each service is */
21 /* identified by a constant string and modeled by a pointer; the latter */
22 /* generally corresponds to a structure containing function pointers. */
24 /* Note that a service's data cannot be a mere function pointer because */
25 /* in C it is possible that function pointers might be implemented */
26 /* differently than data pointers (e.g. 48 bits instead of 32). */
28 /*************************************************************************/
37 #if defined( _MSC_VER ) /* Visual C++ (and Intel C++) */
39 /* we disable the warning `conditional expression is constant' here */
40 /* in order to compile cleanly with the maximum level of warnings */
41 #pragma warning( disable : 4127 )
47 * FT_FACE_FIND_SERVICE
50 * This macro is used to look up a service from a face's driver module.
54 * The source face handle.
57 * A string describing the service as defined in the service's
58 * header files (e.g. FT_SERVICE_ID_MULTI_MASTERS which expands to
59 * `multi-masters'). It is automatically prefixed with
64 * A variable that receives the service pointer. Will be NULL
69 #define FT_FACE_FIND_SERVICE( face, ptr, id ) \
71 FT_Module module = FT_MODULE( FT_FACE( face )->driver ); \
72 FT_Pointer _tmp_ = NULL; \
73 FT_Pointer* _pptr_ = (FT_Pointer*)&(ptr); \
76 if ( module->clazz->get_interface ) \
77 _tmp_ = module->clazz->get_interface( module, FT_SERVICE_ID_ ## id ); \
83 #define FT_FACE_FIND_SERVICE( face, ptr, id ) \
85 FT_Module module = FT_MODULE( FT_FACE( face )->driver ); \
86 FT_Pointer _tmp_ = NULL; \
88 if ( module->clazz->get_interface ) \
89 _tmp_ = module->clazz->get_interface( module, FT_SERVICE_ID_ ## id ); \
97 * FT_FACE_FIND_GLOBAL_SERVICE
100 * This macro is used to look up a service from all modules.
104 * The source face handle.
107 * A string describing the service as defined in the service's
108 * header files (e.g. FT_SERVICE_ID_MULTI_MASTERS which expands to
109 * `multi-masters'). It is automatically prefixed with
114 * A variable that receives the service pointer. Will be NULL
119 #define FT_FACE_FIND_GLOBAL_SERVICE( face, ptr, id ) \
121 FT_Module module = FT_MODULE( FT_FACE( face )->driver ); \
123 FT_Pointer* _pptr_ = (FT_Pointer*)&(ptr); \
126 _tmp_ = ft_module_get_service( module, FT_SERVICE_ID_ ## id ); \
132 #define FT_FACE_FIND_GLOBAL_SERVICE( face, ptr, id ) \
134 FT_Module module = FT_MODULE( FT_FACE( face )->driver ); \
138 _tmp_ = ft_module_get_service( module, FT_SERVICE_ID_ ## id ); \
145 /*************************************************************************/
146 /*************************************************************************/
148 /***** S E R V I C E D E S C R I P T O R S *****/
150 /*************************************************************************/
151 /*************************************************************************/
154 * The following structure is used to _describe_ a given service
155 * to the library. This is useful to build simple static service lists.
157 typedef struct FT_ServiceDescRec_
159 const char* serv_id; /* service name */
160 const void* serv_data; /* service pointer/data */
164 typedef const FT_ServiceDescRec* FT_ServiceDesc;
167 /*************************************************************************/
170 /* FT_DEFINE_SERVICEDESCREC1 .. FT_DEFINE_SERVICEDESCREC6 */
173 /* Used to initialize an array of FT_ServiceDescRec structures. */
175 /* When FT_CONFIG_OPTION_PIC is defined a `create' function needs to */
176 /* be called with a pointer to return an allocated array. As soon as */
177 /* it is no longer needed, a `destroy' function needs to be called to */
178 /* release that allocation. */
180 /* These functions should be manually called from the `pic_init' and */
181 /* `pic_free' functions of your module (see FT_DEFINE_MODULE). */
183 /* When FT_CONFIG_OPTION_PIC is not defined the array will be */
184 /* allocated in the global scope (or the scope where the macro is */
187 #ifndef FT_CONFIG_OPTION_PIC
189 #define FT_DEFINE_SERVICEDESCREC1( class_, \
190 serv_id_1, serv_data_1 ) \
191 static const FT_ServiceDescRec class_[] = \
193 { serv_id_1, serv_data_1 }, \
197 #define FT_DEFINE_SERVICEDESCREC2( class_, \
198 serv_id_1, serv_data_1, \
199 serv_id_2, serv_data_2 ) \
200 static const FT_ServiceDescRec class_[] = \
202 { serv_id_1, serv_data_1 }, \
203 { serv_id_2, serv_data_2 }, \
207 #define FT_DEFINE_SERVICEDESCREC3( class_, \
208 serv_id_1, serv_data_1, \
209 serv_id_2, serv_data_2, \
210 serv_id_3, serv_data_3 ) \
211 static const FT_ServiceDescRec class_[] = \
213 { serv_id_1, serv_data_1 }, \
214 { serv_id_2, serv_data_2 }, \
215 { serv_id_3, serv_data_3 }, \
219 #define FT_DEFINE_SERVICEDESCREC4( class_, \
220 serv_id_1, serv_data_1, \
221 serv_id_2, serv_data_2, \
222 serv_id_3, serv_data_3, \
223 serv_id_4, serv_data_4 ) \
224 static const FT_ServiceDescRec class_[] = \
226 { serv_id_1, serv_data_1 }, \
227 { serv_id_2, serv_data_2 }, \
228 { serv_id_3, serv_data_3 }, \
229 { serv_id_4, serv_data_4 }, \
233 #define FT_DEFINE_SERVICEDESCREC5( class_, \
234 serv_id_1, serv_data_1, \
235 serv_id_2, serv_data_2, \
236 serv_id_3, serv_data_3, \
237 serv_id_4, serv_data_4, \
238 serv_id_5, serv_data_5 ) \
239 static const FT_ServiceDescRec class_[] = \
241 { serv_id_1, serv_data_1 }, \
242 { serv_id_2, serv_data_2 }, \
243 { serv_id_3, serv_data_3 }, \
244 { serv_id_4, serv_data_4 }, \
245 { serv_id_5, serv_data_5 }, \
249 #define FT_DEFINE_SERVICEDESCREC6( class_, \
250 serv_id_1, serv_data_1, \
251 serv_id_2, serv_data_2, \
252 serv_id_3, serv_data_3, \
253 serv_id_4, serv_data_4, \
254 serv_id_5, serv_data_5, \
255 serv_id_6, serv_data_6 ) \
256 static const FT_ServiceDescRec class_[] = \
258 { serv_id_1, serv_data_1 }, \
259 { serv_id_2, serv_data_2 }, \
260 { serv_id_3, serv_data_3 }, \
261 { serv_id_4, serv_data_4 }, \
262 { serv_id_5, serv_data_5 }, \
263 { serv_id_6, serv_data_6 }, \
267 #else /* FT_CONFIG_OPTION_PIC */
269 #define FT_DEFINE_SERVICEDESCREC1( class_, \
270 serv_id_1, serv_data_1 ) \
272 FT_Destroy_Class_ ## class_( FT_Library library, \
273 FT_ServiceDescRec* clazz ) \
275 FT_Memory memory = library->memory; \
283 FT_Create_Class_ ## class_( FT_Library library, \
284 FT_ServiceDescRec** output_class ) \
286 FT_ServiceDescRec* clazz; \
288 FT_Memory memory = library->memory; \
291 if ( FT_ALLOC( clazz, sizeof ( *clazz ) * 2 ) ) \
294 clazz[0].serv_id = serv_id_1; \
295 clazz[0].serv_data = serv_data_1; \
296 clazz[1].serv_id = NULL; \
297 clazz[1].serv_data = NULL; \
299 *output_class = clazz; \
304 #define FT_DEFINE_SERVICEDESCREC2( class_, \
305 serv_id_1, serv_data_1, \
306 serv_id_2, serv_data_2 ) \
308 FT_Destroy_Class_ ## class_( FT_Library library, \
309 FT_ServiceDescRec* clazz ) \
311 FT_Memory memory = library->memory; \
319 FT_Create_Class_ ## class_( FT_Library library, \
320 FT_ServiceDescRec** output_class ) \
322 FT_ServiceDescRec* clazz; \
324 FT_Memory memory = library->memory; \
327 if ( FT_ALLOC( clazz, sizeof ( *clazz ) * 3 ) ) \
330 clazz[0].serv_id = serv_id_1; \
331 clazz[0].serv_data = serv_data_1; \
332 clazz[1].serv_id = serv_id_2; \
333 clazz[1].serv_data = serv_data_2; \
334 clazz[2].serv_id = NULL; \
335 clazz[2].serv_data = NULL; \
337 *output_class = clazz; \
342 #define FT_DEFINE_SERVICEDESCREC3( class_, \
343 serv_id_1, serv_data_1, \
344 serv_id_2, serv_data_2, \
345 serv_id_3, serv_data_3 ) \
347 FT_Destroy_Class_ ## class_( FT_Library library, \
348 FT_ServiceDescRec* clazz ) \
350 FT_Memory memory = library->memory; \
358 FT_Create_Class_ ## class_( FT_Library library, \
359 FT_ServiceDescRec** output_class ) \
361 FT_ServiceDescRec* clazz; \
363 FT_Memory memory = library->memory; \
366 if ( FT_ALLOC( clazz, sizeof ( *clazz ) * 4 ) ) \
369 clazz[0].serv_id = serv_id_1; \
370 clazz[0].serv_data = serv_data_1; \
371 clazz[1].serv_id = serv_id_2; \
372 clazz[1].serv_data = serv_data_2; \
373 clazz[2].serv_id = serv_id_3; \
374 clazz[2].serv_data = serv_data_3; \
375 clazz[3].serv_id = NULL; \
376 clazz[3].serv_data = NULL; \
378 *output_class = clazz; \
383 #define FT_DEFINE_SERVICEDESCREC4( class_, \
384 serv_id_1, serv_data_1, \
385 serv_id_2, serv_data_2, \
386 serv_id_3, serv_data_3, \
387 serv_id_4, serv_data_4 ) \
389 FT_Destroy_Class_ ## class_( FT_Library library, \
390 FT_ServiceDescRec* clazz ) \
392 FT_Memory memory = library->memory; \
400 FT_Create_Class_ ## class_( FT_Library library, \
401 FT_ServiceDescRec** output_class ) \
403 FT_ServiceDescRec* clazz; \
405 FT_Memory memory = library->memory; \
408 if ( FT_ALLOC( clazz, sizeof ( *clazz ) * 5 ) ) \
411 clazz[0].serv_id = serv_id_1; \
412 clazz[0].serv_data = serv_data_1; \
413 clazz[1].serv_id = serv_id_2; \
414 clazz[1].serv_data = serv_data_2; \
415 clazz[2].serv_id = serv_id_3; \
416 clazz[2].serv_data = serv_data_3; \
417 clazz[3].serv_id = serv_id_4; \
418 clazz[3].serv_data = serv_data_4; \
419 clazz[4].serv_id = NULL; \
420 clazz[4].serv_data = NULL; \
422 *output_class = clazz; \
427 #define FT_DEFINE_SERVICEDESCREC5( class_, \
428 serv_id_1, serv_data_1, \
429 serv_id_2, serv_data_2, \
430 serv_id_3, serv_data_3, \
431 serv_id_4, serv_data_4, \
432 serv_id_5, serv_data_5 ) \
434 FT_Destroy_Class_ ## class_( FT_Library library, \
435 FT_ServiceDescRec* clazz ) \
437 FT_Memory memory = library->memory; \
445 FT_Create_Class_ ## class_( FT_Library library, \
446 FT_ServiceDescRec** output_class ) \
448 FT_ServiceDescRec* clazz; \
450 FT_Memory memory = library->memory; \
453 if ( FT_ALLOC( clazz, sizeof ( *clazz ) * 6 ) ) \
456 clazz[0].serv_id = serv_id_1; \
457 clazz[0].serv_data = serv_data_1; \
458 clazz[1].serv_id = serv_id_2; \
459 clazz[1].serv_data = serv_data_2; \
460 clazz[2].serv_id = serv_id_3; \
461 clazz[2].serv_data = serv_data_3; \
462 clazz[3].serv_id = serv_id_4; \
463 clazz[3].serv_data = serv_data_4; \
464 clazz[4].serv_id = serv_id_5; \
465 clazz[4].serv_data = serv_data_5; \
466 clazz[5].serv_id = NULL; \
467 clazz[5].serv_data = NULL; \
469 *output_class = clazz; \
474 #define FT_DEFINE_SERVICEDESCREC6( class_, \
475 serv_id_1, serv_data_1, \
476 serv_id_2, serv_data_2, \
477 serv_id_3, serv_data_3, \
478 serv_id_4, serv_data_4, \
479 serv_id_5, serv_data_5, \
480 serv_id_6, serv_data_6 ) \
482 FT_Destroy_Class_ ## class_( FT_Library library, \
483 FT_ServiceDescRec* clazz ) \
485 FT_Memory memory = library->memory; \
493 FT_Create_Class_ ## class_( FT_Library library, \
494 FT_ServiceDescRec** output_class) \
496 FT_ServiceDescRec* clazz; \
498 FT_Memory memory = library->memory; \
501 if ( FT_ALLOC( clazz, sizeof ( *clazz ) * 7 ) ) \
504 clazz[0].serv_id = serv_id_1; \
505 clazz[0].serv_data = serv_data_1; \
506 clazz[1].serv_id = serv_id_2; \
507 clazz[1].serv_data = serv_data_2; \
508 clazz[2].serv_id = serv_id_3; \
509 clazz[2].serv_data = serv_data_3; \
510 clazz[3].serv_id = serv_id_4; \
511 clazz[3].serv_data = serv_data_4; \
512 clazz[4].serv_id = serv_id_5; \
513 clazz[4].serv_data = serv_data_5; \
514 clazz[5].serv_id = serv_id_6; \
515 clazz[5].serv_data = serv_data_6; \
516 clazz[6].serv_id = NULL; \
517 clazz[6].serv_data = NULL; \
519 *output_class = clazz; \
524 #endif /* FT_CONFIG_OPTION_PIC */
528 * Parse a list of FT_ServiceDescRec descriptors and look for
529 * a specific service by ID. Note that the last element in the
530 * array must be { NULL, NULL }, and that the function should
531 * return NULL if the service isn't available.
533 * This function can be used by modules to implement their
534 * `get_service' method.
536 FT_BASE( FT_Pointer )
537 ft_service_list_lookup( FT_ServiceDesc service_descriptors,
538 const char* service_id );
541 /*************************************************************************/
542 /*************************************************************************/
544 /***** S E R V I C E S C A C H E *****/
546 /*************************************************************************/
547 /*************************************************************************/
550 * This structure is used to store a cache for several frequently used
551 * services. It is the type of `face->internal->services'. You
552 * should only use FT_FACE_LOOKUP_SERVICE to access it.
554 * All fields should have the type FT_Pointer to relax compilation
555 * dependencies. We assume the developer isn't completely stupid.
557 * Each field must be named `service_XXXX' where `XXX' corresponds to
558 * the correct FT_SERVICE_ID_XXXX macro. See the definition of
559 * FT_FACE_LOOKUP_SERVICE below how this is implemented.
562 typedef struct FT_ServiceCacheRec_
564 FT_Pointer service_POSTSCRIPT_FONT_NAME;
565 FT_Pointer service_MULTI_MASTERS;
566 FT_Pointer service_GLYPH_DICT;
567 FT_Pointer service_PFR_METRICS;
568 FT_Pointer service_WINFNT;
570 } FT_ServiceCacheRec, *FT_ServiceCache;
574 * A magic number used within the services cache.
576 #define FT_SERVICE_UNAVAILABLE ((FT_Pointer)-2) /* magic number */
581 * FT_FACE_LOOKUP_SERVICE
584 * This macro is used to lookup a service from a face's driver module
589 * The source face handle containing the cache.
592 * The field name in the cache.
599 * A variable receiving the service data. NULL if not available.
603 #define FT_FACE_LOOKUP_SERVICE( face, ptr, id ) \
606 FT_Pointer* Pptr = (FT_Pointer*)&(ptr); \
609 svc = FT_FACE( face )->internal->services. service_ ## id; \
610 if ( svc == FT_SERVICE_UNAVAILABLE ) \
612 else if ( svc == NULL ) \
614 FT_FACE_FIND_SERVICE( face, svc, id ); \
616 FT_FACE( face )->internal->services. service_ ## id = \
617 (FT_Pointer)( svc != NULL ? svc \
618 : FT_SERVICE_UNAVAILABLE ); \
625 #define FT_FACE_LOOKUP_SERVICE( face, ptr, id ) \
630 svc = FT_FACE( face )->internal->services. service_ ## id; \
631 if ( svc == FT_SERVICE_UNAVAILABLE ) \
633 else if ( svc == NULL ) \
635 FT_FACE_FIND_SERVICE( face, svc, id ); \
637 FT_FACE( face )->internal->services. service_ ## id = \
638 (FT_Pointer)( svc != NULL ? svc \
639 : FT_SERVICE_UNAVAILABLE ); \
647 * A macro used to define new service structure types.
650 #define FT_DEFINE_SERVICE( name ) \
651 typedef struct FT_Service_ ## name ## Rec_ \
652 FT_Service_ ## name ## Rec ; \
653 typedef struct FT_Service_ ## name ## Rec_ \
654 const * FT_Service_ ## name ; \
655 struct FT_Service_ ## name ## Rec_
660 * The header files containing the services.
663 #define FT_SERVICE_BDF_H <freetype/internal/services/svbdf.h>
664 #define FT_SERVICE_CID_H <freetype/internal/services/svcid.h>
665 #define FT_SERVICE_GLYPH_DICT_H <freetype/internal/services/svgldict.h>
666 #define FT_SERVICE_GX_VALIDATE_H <freetype/internal/services/svgxval.h>
667 #define FT_SERVICE_KERNING_H <freetype/internal/services/svkern.h>
668 #define FT_SERVICE_MULTIPLE_MASTERS_H <freetype/internal/services/svmm.h>
669 #define FT_SERVICE_OPENTYPE_VALIDATE_H <freetype/internal/services/svotval.h>
670 #define FT_SERVICE_PFR_H <freetype/internal/services/svpfr.h>
671 #define FT_SERVICE_POSTSCRIPT_CMAPS_H <freetype/internal/services/svpscmap.h>
672 #define FT_SERVICE_POSTSCRIPT_INFO_H <freetype/internal/services/svpsinfo.h>
673 #define FT_SERVICE_POSTSCRIPT_NAME_H <freetype/internal/services/svpostnm.h>
674 #define FT_SERVICE_SFNT_H <freetype/internal/services/svsfnt.h>
675 #define FT_SERVICE_TRUETYPE_ENGINE_H <freetype/internal/services/svtteng.h>
676 #define FT_SERVICE_TT_CMAP_H <freetype/internal/services/svttcmap.h>
677 #define FT_SERVICE_WINFNT_H <freetype/internal/services/svwinfnt.h>
678 #define FT_SERVICE_XFREE86_NAME_H <freetype/internal/services/svxf86nm.h>
679 #define FT_SERVICE_TRUETYPE_GLYF_H <freetype/internal/services/svttglyf.h>
685 #endif /* __FTSERV_H__ */