2000-08-23 19:32:42 +02:00
|
|
|
/***************************************************************************/
|
|
|
|
/* */
|
2000-08-24 18:29:15 +02:00
|
|
|
/* ftcmanag.c */
|
2000-08-23 19:32:42 +02:00
|
|
|
/* */
|
2000-08-24 18:29:15 +02:00
|
|
|
/* FreeType Cache Manager (body). */
|
2000-08-23 19:32:42 +02:00
|
|
|
/* */
|
2001-06-28 19:49:10 +02:00
|
|
|
/* Copyright 2000-2001 by */
|
2000-08-23 19:32:42 +02:00
|
|
|
/* David Turner, Robert Wilhelm, and Werner Lemberg. */
|
|
|
|
/* */
|
|
|
|
/* This file is part of the FreeType project, and may only be used, */
|
|
|
|
/* modified, and distributed under the terms of the FreeType project */
|
|
|
|
/* license, LICENSE.TXT. By continuing to use, modify, or distribute */
|
|
|
|
/* this file you indicate that you have read the license and */
|
|
|
|
/* understand and accept it fully. */
|
|
|
|
/* */
|
|
|
|
/***************************************************************************/
|
|
|
|
|
2000-12-08 17:17:16 +01:00
|
|
|
|
|
|
|
#include <ft2build.h>
|
|
|
|
#include FT_CACHE_H
|
|
|
|
#include FT_CACHE_MANAGER_H
|
2001-10-26 18:58:27 +02:00
|
|
|
#include FT_CACHE_INTERNAL_LRU_H
|
2000-12-08 17:17:16 +01:00
|
|
|
#include FT_INTERNAL_OBJECTS_H
|
|
|
|
#include FT_INTERNAL_DEBUG_H
|
|
|
|
#include FT_LIST_H
|
2001-10-07 15:30:26 +02:00
|
|
|
#include FT_SIZES_H
|
2000-08-23 13:22:30 +02:00
|
|
|
|
Complete redesign of error codes. Please check ftmoderr.h for more
details.
* include/freetype/internal/cfferrs.h,
include/freetype/internal/tterrors.h,
include/freetype/internal/t1errors.h: Removed. Replaced with files
local to the module. All extra error codes have been moved to
`fterrors.h'.
* src/sfnt/ttpost.h: Move error codes to `fterrors.h'.
* src/autohint/aherrors.h, src/cache/ftcerror.h, src/cff/cfferrs.h,
src/cid/ciderrs.h, src/pcf/pcferror.h, src/psaux/psauxerr.h,
src/psnames/psnamerr.h, src/raster/rasterrs.h, src/sfnt/sferrors.h,
src/smooth/ftsmerrs.h, src/truetype/tterrors.h,
src/type1/t1errors.h, src/winfonts/fnterrs.h: New files defining the
error names for the module it belongs to.
* include/freetype/ftmoderr.h: New file, defining the module error
offsets. Its structure is similar to `fterrors.h'.
* include/freetype/fterrors.h (FT_NOERRORDEF): New macro.
(FT_ERRORDEF): Redefined to use module error offsets.
All internal error codes are now public; unused error codes have
been removed, some are new.
* include/freetype/config/ftheader.h (FT_MODULE_ERRORS_H): New
macro.
* include/freetype/config/ftoption.h
(FT_CONFIG_OPTION_USE_MODULE_ERRORS): New macro.
All other source files have been updated to use the new error codes;
some already existing (internal) error codes local to a module have
been renamed to give them the same name as in the base module.
All make files have been updated to include the local error files.
* src/cid/cidtokens.h: Replaced with...
* src/cid/cidtoken.h: This file for 8+3 consistency.
* src/raster/ftraster.c: Use macros for header file names.
2001-06-06 19:30:41 +02:00
|
|
|
#include "ftcerror.h"
|
|
|
|
|
2000-10-12 07:05:40 +02:00
|
|
|
|
|
|
|
#undef FT_COMPONENT
|
|
|
|
#define FT_COMPONENT trace_cache
|
2000-08-23 13:22:30 +02:00
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
#define FTC_LRU_GET_MANAGER( lru ) ( (FTC_Manager)(lru)->user_data )
|
2000-08-23 19:32:42 +02:00
|
|
|
|
|
|
|
|
2000-08-29 18:50:01 +02:00
|
|
|
/*************************************************************************/
|
|
|
|
/*************************************************************************/
|
|
|
|
/***** *****/
|
2001-10-26 18:58:27 +02:00
|
|
|
/***** FACE LRU IMPLEMENTATION *****/
|
2000-08-29 18:50:01 +02:00
|
|
|
/***** *****/
|
|
|
|
/*************************************************************************/
|
|
|
|
/*************************************************************************/
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
typedef struct FTC_FaceNodeRec_* FTC_FaceNode;
|
|
|
|
typedef struct FTC_SizeNodeRec_* FTC_SizeNode;
|
2001-10-26 18:58:27 +02:00
|
|
|
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
typedef struct FTC_FaceNodeRec_
|
2001-10-26 18:58:27 +02:00
|
|
|
{
|
|
|
|
FT_LruNodeRec lru;
|
|
|
|
FT_Face face;
|
|
|
|
|
|
|
|
} FTC_FaceNodeRec;
|
|
|
|
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
typedef struct FTC_SizeNodeRec_
|
2001-10-26 18:58:27 +02:00
|
|
|
{
|
|
|
|
FT_LruNodeRec lru;
|
|
|
|
FT_Size size;
|
|
|
|
|
|
|
|
} FTC_SizeNodeRec;
|
|
|
|
|
|
|
|
|
2001-06-27 18:18:10 +02:00
|
|
|
FT_CALLBACK_DEF( FT_Error )
|
2001-12-05 02:22:05 +01:00
|
|
|
ftc_face_node_init( FTC_FaceNode node,
|
|
|
|
FTC_FaceID face_id,
|
|
|
|
FT_LruList list )
|
2000-08-23 13:22:30 +02:00
|
|
|
{
|
2001-12-05 02:22:05 +01:00
|
|
|
FTC_Manager manager = FTC_LRU_GET_MANAGER( list );
|
2000-08-23 13:22:30 +02:00
|
|
|
FT_Error error;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-08-23 19:32:42 +02:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
error = manager->request_face( face_id,
|
2000-08-23 23:11:13 +02:00
|
|
|
manager->library,
|
2000-08-23 13:22:30 +02:00
|
|
|
manager->request_data,
|
2001-10-26 18:58:27 +02:00
|
|
|
&node->face );
|
2000-08-23 19:32:42 +02:00
|
|
|
if ( !error )
|
2000-08-23 13:22:30 +02:00
|
|
|
{
|
2000-08-23 19:32:42 +02:00
|
|
|
/* destroy initial size object; it will be re-created later */
|
2001-10-26 18:58:27 +02:00
|
|
|
if ( node->face->size )
|
|
|
|
FT_Done_Size( node->face->size );
|
2000-08-23 13:22:30 +02:00
|
|
|
}
|
2000-08-23 19:32:42 +02:00
|
|
|
|
2000-08-23 13:22:30 +02:00
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
/* helper function for ftc_manager_done_face() */
|
|
|
|
FT_CALLBACK_DEF( FT_Bool )
|
|
|
|
ftc_size_node_select( FTC_SizeNode node,
|
|
|
|
FT_Face face )
|
|
|
|
{
|
|
|
|
return FT_BOOL( node->size->face == face );
|
|
|
|
}
|
2000-08-23 19:32:42 +02:00
|
|
|
|
2000-08-23 13:22:30 +02:00
|
|
|
|
2001-06-27 18:18:10 +02:00
|
|
|
FT_CALLBACK_DEF( void )
|
2001-10-26 18:58:27 +02:00
|
|
|
ftc_face_node_done( FTC_FaceNode node,
|
|
|
|
FT_LruList list )
|
2000-08-23 13:22:30 +02:00
|
|
|
{
|
2001-12-05 02:22:05 +01:00
|
|
|
FTC_Manager manager = FTC_LRU_GET_MANAGER( list );
|
2001-10-26 18:58:27 +02:00
|
|
|
FT_Face face = node->face;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-08-23 19:32:42 +02:00
|
|
|
|
2000-08-23 13:22:30 +02:00
|
|
|
/* we must begin by removing all sizes for the target face */
|
2000-08-23 19:32:42 +02:00
|
|
|
/* from the manager's list */
|
2001-10-26 18:58:27 +02:00
|
|
|
FT_LruList_Remove_Selection( manager->sizes_list,
|
2001-12-05 02:22:05 +01:00
|
|
|
(FT_LruNode_SelectFunc)ftc_size_node_select,
|
2001-10-26 18:58:27 +02:00
|
|
|
face );
|
2000-08-23 13:22:30 +02:00
|
|
|
|
2000-10-12 07:05:40 +02:00
|
|
|
/* all right, we can discard the face now */
|
2000-08-23 13:22:30 +02:00
|
|
|
FT_Done_Face( face );
|
2001-10-26 18:58:27 +02:00
|
|
|
node->face = NULL;
|
2000-08-23 13:22:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
FT_CALLBACK_TABLE_DEF
|
2001-12-05 02:22:05 +01:00
|
|
|
const FT_LruList_ClassRec ftc_face_list_class =
|
2001-10-26 18:58:27 +02:00
|
|
|
{
|
2001-12-05 02:22:05 +01:00
|
|
|
sizeof ( FT_LruListRec ),
|
|
|
|
(FT_LruList_InitFunc) 0,
|
|
|
|
(FT_LruList_DoneFunc) 0,
|
|
|
|
|
|
|
|
sizeof ( FTC_FaceNodeRec ),
|
|
|
|
(FT_LruNode_InitFunc) ftc_face_node_init,
|
|
|
|
(FT_LruNode_DoneFunc) ftc_face_node_done,
|
|
|
|
(FT_LruNode_FlushFunc) 0, /* no flushing needed */
|
|
|
|
(FT_LruNode_CompareFunc)0, /* direct comparison of FTC_FaceID handles */
|
2001-10-26 18:58:27 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/*************************************************************************/
|
|
|
|
/*************************************************************************/
|
|
|
|
/***** *****/
|
|
|
|
/***** SIZES LRU IMPLEMENTATION *****/
|
|
|
|
/***** *****/
|
|
|
|
/*************************************************************************/
|
|
|
|
/*************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
typedef struct FTC_SizeQueryRec_
|
2000-08-23 13:22:30 +02:00
|
|
|
{
|
2001-10-26 18:58:27 +02:00
|
|
|
FT_Face face;
|
|
|
|
FT_UInt width;
|
|
|
|
FT_UInt height;
|
|
|
|
|
|
|
|
} FTC_SizeQueryRec, *FTC_SizeQuery;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-08-23 13:22:30 +02:00
|
|
|
|
2001-06-27 18:18:10 +02:00
|
|
|
FT_CALLBACK_DEF( FT_Error )
|
2001-10-26 18:58:27 +02:00
|
|
|
ftc_size_node_init( FTC_SizeNode node,
|
|
|
|
FTC_SizeQuery query )
|
2000-08-23 13:22:30 +02:00
|
|
|
{
|
2001-10-26 18:58:27 +02:00
|
|
|
FT_Face face = query->face;
|
|
|
|
FT_Size size;
|
|
|
|
FT_Error error;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-08-23 19:32:42 +02:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
node->size = NULL;
|
2000-08-24 13:53:35 +02:00
|
|
|
error = FT_New_Size( face, &size );
|
2000-08-23 19:32:42 +02:00
|
|
|
if ( !error )
|
2000-08-23 13:22:30 +02:00
|
|
|
{
|
2001-10-07 15:30:26 +02:00
|
|
|
FT_Activate_Size( size );
|
2001-10-26 18:58:27 +02:00
|
|
|
error = FT_Set_Pixel_Sizes( query->face,
|
|
|
|
query->width,
|
|
|
|
query->height );
|
2000-08-23 19:32:42 +02:00
|
|
|
if ( error )
|
|
|
|
FT_Done_Size( size );
|
2000-08-23 13:22:30 +02:00
|
|
|
else
|
2001-10-26 18:58:27 +02:00
|
|
|
node->size = size;
|
2000-08-23 13:22:30 +02:00
|
|
|
}
|
2000-10-12 07:05:40 +02:00
|
|
|
return error;
|
2000-08-23 13:22:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2001-06-27 18:18:10 +02:00
|
|
|
FT_CALLBACK_DEF( void )
|
2001-10-26 18:58:27 +02:00
|
|
|
ftc_size_node_done( FTC_SizeNode node )
|
2000-08-23 13:22:30 +02:00
|
|
|
{
|
2001-12-05 02:22:05 +01:00
|
|
|
if ( node->size )
|
2001-10-26 18:58:27 +02:00
|
|
|
{
|
|
|
|
FT_Done_Size( node->size );
|
|
|
|
node->size = NULL;
|
|
|
|
}
|
2000-10-12 07:05:40 +02:00
|
|
|
}
|
2000-08-23 13:22:30 +02:00
|
|
|
|
|
|
|
|
2001-06-27 18:18:10 +02:00
|
|
|
FT_CALLBACK_DEF( FT_Error )
|
2001-10-26 18:58:27 +02:00
|
|
|
ftc_size_node_flush( FTC_SizeNode node,
|
|
|
|
FTC_SizeQuery query )
|
2000-08-23 13:22:30 +02:00
|
|
|
{
|
2001-10-26 18:58:27 +02:00
|
|
|
FT_Size size = node->size;
|
|
|
|
FT_Error error;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-08-23 19:32:42 +02:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
if ( size->face == query->face )
|
2000-08-23 13:22:30 +02:00
|
|
|
{
|
Fixed a bug in `glnames.py' that prevented it from generating
correct glyph names tables. This resulted in the unavailability of
certain glyphs like `Cacute', `cacute' and `lslash' in Unicode
charmaps, even if these were present in the font (causing problems
for Polish users).
* src/tools/glnames.py (mac_standard_names): Fixed.
(t1_standard_strings): Some fixes and renamed to ...
(sid_standard_names): This.
(t1_expert_encoding): Fixed.
(the_adobe_glyph_list): Renamed to ...
(adobe_glyph_names): This.
(the_adobe_glyphs): Renamed to ...
(adobe_glyph_values): This.
(dump_mac_indices, dump_glyph_list, dump_unicode_values, main):
Updated.
* src/psnames/pstables.h: Regenerated.
* src/psnames/psmodule.c (PS_Unicode_Value): Fix offset.
Fix return value.
Use `sid_standard_table' and `ps_names_to_unicode' instead of
`t1_standard_glyphs' and `names_to_unicode'.
(PS_Macintosh_Name): Use `ps_glyph_names' instead of
`standard_glyph_names'.
(PS_Standard_Strings): Use `sid_standard_names' instead of
`t1_standard_glyphs'.
* doc/BUGS, doc/TODO: New documents.
* src/cache/ftlru.c (FT_Lru_Lookup_Node): Fixed a bug that prevented
correct LRU behaviour.
setjmp() and longjmp() are now used for rollback (i.e. when memory
pool overflow occurs).
Function names are now all uniformly prefixed with `gray_'.
* src/smooth/ftgrays.c: Include <setjmp.h>.
(ErrRaster_MemoryOverflow): New macro.
(TArea): New type to store area values in each cell (using `int' was
too small on 16-bit systems). <limits.h> is included to properly
get the needed data type.
(TCell, TRaster): Use it.
(TRaster): New element `jump_buffer'.
(gray_compute_cbox): Use `RAS_ARG' as the only parameter and get
`outline' from it.
(gray_record_cell): Use longjmp().
(gray_set_cell): Use gray_record_cell() for error handling.
(gray_render_line, gray_render_conic, gray_render_cubic): Simplify.
(gray_convert_glyph_inner): New function, using setjmp().
(gray_convert_glyph): Use it.
Provide a public API to manage multiple size objects for a given
FT_Face in the new header file `ftsizes.h'.
* include/freetype/ftsizes.h: New header file,
* include/freetype/internal/ftobjs.h: Use it.
Remove declarations of FT_New_Size and FT_Done_Size (moved to
ftsizes.h).
* include/freetype/config/ftheader.h (FT_SIZES_H): New macro.
* src/base/ftobjs.c (FT_Activate_Size): New function.
* src/cache/ftcmanag.c: Include ftsizes.h.
(ftc_manager_init_size, ftc_manager_flush_size): Use
FT_Activate_Size.
2001-10-10 21:56:42 +02:00
|
|
|
FT_Activate_Size( size );
|
2001-10-26 18:58:27 +02:00
|
|
|
error = FT_Set_Pixel_Sizes( query->face, query->width, query->height );
|
2000-08-23 19:32:42 +02:00
|
|
|
if ( error )
|
2001-10-26 18:58:27 +02:00
|
|
|
{
|
2000-08-23 13:22:30 +02:00
|
|
|
FT_Done_Size( size );
|
2001-10-26 18:58:27 +02:00
|
|
|
node->size = NULL;
|
|
|
|
}
|
2000-08-23 13:22:30 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2000-08-23 19:32:42 +02:00
|
|
|
FT_Done_Size( size );
|
2001-10-26 18:58:27 +02:00
|
|
|
node->size = NULL;
|
|
|
|
|
|
|
|
error = ftc_size_node_init( node, query );
|
2000-08-23 13:22:30 +02:00
|
|
|
}
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2001-06-27 18:18:10 +02:00
|
|
|
FT_CALLBACK_DEF( FT_Bool )
|
2001-10-26 18:58:27 +02:00
|
|
|
ftc_size_node_compare( FTC_SizeNode node,
|
|
|
|
FTC_SizeQuery query )
|
2000-08-23 13:22:30 +02:00
|
|
|
{
|
2001-10-26 18:58:27 +02:00
|
|
|
FT_Size size = node->size;
|
2000-08-23 13:22:30 +02:00
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
|
|
|
|
return FT_BOOL( size->face == query->face &&
|
|
|
|
(FT_UInt)size->metrics.x_ppem == query->width &&
|
|
|
|
(FT_UInt)size->metrics.y_ppem == query->height );
|
2000-08-23 19:32:42 +02:00
|
|
|
}
|
2000-08-23 13:22:30 +02:00
|
|
|
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-11-04 02:55:49 +01:00
|
|
|
FT_CALLBACK_TABLE_DEF
|
2001-10-26 18:58:27 +02:00
|
|
|
const FT_LruList_ClassRec ftc_size_list_class =
|
2000-08-23 13:22:30 +02:00
|
|
|
{
|
2001-10-26 18:58:27 +02:00
|
|
|
sizeof ( FT_LruListRec ),
|
2001-12-05 02:22:05 +01:00
|
|
|
(FT_LruList_InitFunc) 0,
|
|
|
|
(FT_LruList_DoneFunc) 0,
|
|
|
|
|
|
|
|
sizeof ( FTC_SizeNodeRec ),
|
|
|
|
(FT_LruNode_InitFunc) ftc_size_node_init,
|
|
|
|
(FT_LruNode_DoneFunc) ftc_size_node_done,
|
|
|
|
(FT_LruNode_FlushFunc) ftc_size_node_flush,
|
|
|
|
(FT_LruNode_CompareFunc)ftc_size_node_compare
|
2000-08-23 13:22:30 +02:00
|
|
|
};
|
|
|
|
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
/*************************************************************************/
|
|
|
|
/*************************************************************************/
|
|
|
|
/***** *****/
|
|
|
|
/***** CACHE MANAGER ROUTINES *****/
|
|
|
|
/***** *****/
|
|
|
|
/*************************************************************************/
|
|
|
|
/*************************************************************************/
|
2000-08-23 13:22:30 +02:00
|
|
|
|
|
|
|
|
2001-06-28 19:49:10 +02:00
|
|
|
/* documentation is in ftcache.h */
|
|
|
|
|
2001-06-27 18:18:10 +02:00
|
|
|
FT_EXPORT_DEF( FT_Error )
|
|
|
|
FTC_Manager_New( FT_Library library,
|
|
|
|
FT_UInt max_faces,
|
|
|
|
FT_UInt max_sizes,
|
|
|
|
FT_ULong max_bytes,
|
|
|
|
FTC_Face_Requester requester,
|
|
|
|
FT_Pointer req_data,
|
|
|
|
FTC_Manager *amanager )
|
2000-08-23 13:22:30 +02:00
|
|
|
{
|
|
|
|
FT_Error error;
|
2000-08-24 18:29:15 +02:00
|
|
|
FT_Memory memory;
|
2000-08-23 13:22:30 +02:00
|
|
|
FTC_Manager manager = 0;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
|
|
|
|
2000-08-24 18:29:15 +02:00
|
|
|
if ( !library )
|
Complete redesign of error codes. Please check ftmoderr.h for more
details.
* include/freetype/internal/cfferrs.h,
include/freetype/internal/tterrors.h,
include/freetype/internal/t1errors.h: Removed. Replaced with files
local to the module. All extra error codes have been moved to
`fterrors.h'.
* src/sfnt/ttpost.h: Move error codes to `fterrors.h'.
* src/autohint/aherrors.h, src/cache/ftcerror.h, src/cff/cfferrs.h,
src/cid/ciderrs.h, src/pcf/pcferror.h, src/psaux/psauxerr.h,
src/psnames/psnamerr.h, src/raster/rasterrs.h, src/sfnt/sferrors.h,
src/smooth/ftsmerrs.h, src/truetype/tterrors.h,
src/type1/t1errors.h, src/winfonts/fnterrs.h: New files defining the
error names for the module it belongs to.
* include/freetype/ftmoderr.h: New file, defining the module error
offsets. Its structure is similar to `fterrors.h'.
* include/freetype/fterrors.h (FT_NOERRORDEF): New macro.
(FT_ERRORDEF): Redefined to use module error offsets.
All internal error codes are now public; unused error codes have
been removed, some are new.
* include/freetype/config/ftheader.h (FT_MODULE_ERRORS_H): New
macro.
* include/freetype/config/ftoption.h
(FT_CONFIG_OPTION_USE_MODULE_ERRORS): New macro.
All other source files have been updated to use the new error codes;
some already existing (internal) error codes local to a module have
been renamed to give them the same name as in the base module.
All make files have been updated to include the local error files.
* src/cid/cidtokens.h: Replaced with...
* src/cid/cidtoken.h: This file for 8+3 consistency.
* src/raster/ftraster.c: Use macros for header file names.
2001-06-06 19:30:41 +02:00
|
|
|
return FTC_Err_Invalid_Library_Handle;
|
2000-08-24 18:29:15 +02:00
|
|
|
|
|
|
|
memory = library->memory;
|
|
|
|
|
2000-08-23 19:32:42 +02:00
|
|
|
if ( ALLOC( manager, sizeof ( *manager ) ) )
|
2000-08-23 13:22:30 +02:00
|
|
|
goto Exit;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-08-24 18:29:15 +02:00
|
|
|
if ( max_faces == 0 )
|
2000-09-19 03:11:11 +02:00
|
|
|
max_faces = FTC_MAX_FACES_DEFAULT;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-08-24 18:29:15 +02:00
|
|
|
if ( max_sizes == 0 )
|
2000-09-19 03:11:11 +02:00
|
|
|
max_sizes = FTC_MAX_SIZES_DEFAULT;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-09-19 03:11:11 +02:00
|
|
|
if ( max_bytes == 0 )
|
|
|
|
max_bytes = FTC_MAX_BYTES_DEFAULT;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
error = FT_LruList_New( &ftc_face_list_class,
|
|
|
|
max_faces,
|
|
|
|
manager,
|
|
|
|
memory,
|
|
|
|
&manager->faces_list );
|
2000-08-23 19:32:42 +02:00
|
|
|
if ( error )
|
2000-08-23 13:22:30 +02:00
|
|
|
goto Exit;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
error = FT_LruList_New( &ftc_size_list_class,
|
|
|
|
max_sizes,
|
|
|
|
manager,
|
|
|
|
memory,
|
|
|
|
&manager->sizes_list );
|
2000-08-23 19:32:42 +02:00
|
|
|
if ( error )
|
2000-08-23 13:22:30 +02:00
|
|
|
goto Exit;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-08-23 13:22:30 +02:00
|
|
|
manager->library = library;
|
2001-10-26 18:58:27 +02:00
|
|
|
manager->max_weight = max_bytes;
|
|
|
|
manager->cur_weight = 0;
|
2001-10-29 11:45:57 +01:00
|
|
|
|
2000-08-23 13:22:30 +02:00
|
|
|
manager->request_face = requester;
|
|
|
|
manager->request_data = req_data;
|
2001-10-07 15:30:26 +02:00
|
|
|
|
2000-08-23 13:22:30 +02:00
|
|
|
*amanager = manager;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-08-23 13:22:30 +02:00
|
|
|
Exit:
|
2000-08-23 19:32:42 +02:00
|
|
|
if ( error && manager )
|
2000-08-23 13:22:30 +02:00
|
|
|
{
|
2001-10-26 18:58:27 +02:00
|
|
|
FT_LruList_Destroy( manager->faces_list );
|
|
|
|
FT_LruList_Destroy( manager->sizes_list );
|
2000-08-23 13:22:30 +02:00
|
|
|
FREE( manager );
|
|
|
|
}
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-08-23 13:22:30 +02:00
|
|
|
return error;
|
|
|
|
}
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-08-23 19:32:42 +02:00
|
|
|
|
2001-06-28 19:49:10 +02:00
|
|
|
/* documentation is in ftcache.h */
|
|
|
|
|
2001-06-27 18:18:10 +02:00
|
|
|
FT_EXPORT_DEF( void )
|
|
|
|
FTC_Manager_Done( FTC_Manager manager )
|
2000-08-23 13:22:30 +02:00
|
|
|
{
|
2000-08-24 18:29:15 +02:00
|
|
|
FT_Memory memory;
|
2000-09-19 03:11:11 +02:00
|
|
|
FT_UInt index;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-08-23 19:32:42 +02:00
|
|
|
|
2000-08-24 18:29:15 +02:00
|
|
|
if ( !manager || !manager->library )
|
|
|
|
return;
|
|
|
|
|
|
|
|
memory = manager->library->memory;
|
|
|
|
|
2000-09-19 03:11:11 +02:00
|
|
|
/* now discard all caches */
|
|
|
|
for (index = 0; index < FTC_MAX_CACHES; index++ )
|
|
|
|
{
|
|
|
|
FTC_Cache cache = manager->caches[index];
|
2000-10-12 07:05:40 +02:00
|
|
|
|
|
|
|
|
|
|
|
if ( cache )
|
2000-09-19 03:11:11 +02:00
|
|
|
{
|
2001-10-26 18:58:27 +02:00
|
|
|
cache->clazz->cache_done( cache );
|
2000-10-12 07:05:40 +02:00
|
|
|
FREE( cache );
|
2000-09-19 03:11:11 +02:00
|
|
|
manager->caches[index] = 0;
|
|
|
|
}
|
|
|
|
}
|
2001-10-29 11:45:57 +01:00
|
|
|
|
2000-09-19 03:11:11 +02:00
|
|
|
/* discard faces and sizes */
|
2001-10-26 18:58:27 +02:00
|
|
|
FT_LruList_Destroy( manager->faces_list );
|
|
|
|
manager->faces_list = 0;
|
2001-06-28 19:49:10 +02:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
FT_LruList_Destroy( manager->sizes_list );
|
|
|
|
manager->sizes_list = 0;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-08-23 13:22:30 +02:00
|
|
|
FREE( manager );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2001-06-28 19:49:10 +02:00
|
|
|
/* documentation is in ftcache.h */
|
|
|
|
|
2001-06-27 18:18:10 +02:00
|
|
|
FT_EXPORT_DEF( void )
|
|
|
|
FTC_Manager_Reset( FTC_Manager manager )
|
2000-08-23 13:22:30 +02:00
|
|
|
{
|
2001-12-05 02:22:05 +01:00
|
|
|
if ( manager )
|
2000-09-19 03:11:11 +02:00
|
|
|
{
|
2001-10-26 18:58:27 +02:00
|
|
|
FT_LruList_Reset( manager->sizes_list );
|
|
|
|
FT_LruList_Reset( manager->faces_list );
|
2000-09-19 03:11:11 +02:00
|
|
|
}
|
2000-10-31 23:13:54 +01:00
|
|
|
/* XXX: FIXME: flush the caches? */
|
2000-08-23 13:22:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2001-06-28 19:49:10 +02:00
|
|
|
/* documentation is in ftcache.h */
|
|
|
|
|
2001-06-27 18:18:10 +02:00
|
|
|
FT_EXPORT_DEF( FT_Error )
|
|
|
|
FTC_Manager_Lookup_Face( FTC_Manager manager,
|
|
|
|
FTC_FaceID face_id,
|
|
|
|
FT_Face *aface )
|
2000-08-23 13:22:30 +02:00
|
|
|
{
|
2001-10-26 18:58:27 +02:00
|
|
|
FT_Error error;
|
|
|
|
FTC_FaceNode node;
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
if ( aface == NULL )
|
|
|
|
return FTC_Err_Bad_Argument;
|
|
|
|
|
|
|
|
*aface = NULL;
|
|
|
|
|
2000-08-24 18:29:15 +02:00
|
|
|
if ( !manager )
|
Complete redesign of error codes. Please check ftmoderr.h for more
details.
* include/freetype/internal/cfferrs.h,
include/freetype/internal/tterrors.h,
include/freetype/internal/t1errors.h: Removed. Replaced with files
local to the module. All extra error codes have been moved to
`fterrors.h'.
* src/sfnt/ttpost.h: Move error codes to `fterrors.h'.
* src/autohint/aherrors.h, src/cache/ftcerror.h, src/cff/cfferrs.h,
src/cid/ciderrs.h, src/pcf/pcferror.h, src/psaux/psauxerr.h,
src/psnames/psnamerr.h, src/raster/rasterrs.h, src/sfnt/sferrors.h,
src/smooth/ftsmerrs.h, src/truetype/tterrors.h,
src/type1/t1errors.h, src/winfonts/fnterrs.h: New files defining the
error names for the module it belongs to.
* include/freetype/ftmoderr.h: New file, defining the module error
offsets. Its structure is similar to `fterrors.h'.
* include/freetype/fterrors.h (FT_NOERRORDEF): New macro.
(FT_ERRORDEF): Redefined to use module error offsets.
All internal error codes are now public; unused error codes have
been removed, some are new.
* include/freetype/config/ftheader.h (FT_MODULE_ERRORS_H): New
macro.
* include/freetype/config/ftoption.h
(FT_CONFIG_OPTION_USE_MODULE_ERRORS): New macro.
All other source files have been updated to use the new error codes;
some already existing (internal) error codes local to a module have
been renamed to give them the same name as in the base module.
All make files have been updated to include the local error files.
* src/cid/cidtokens.h: Replaced with...
* src/cid/cidtoken.h: This file for 8+3 consistency.
* src/raster/ftraster.c: Use macros for header file names.
2001-06-06 19:30:41 +02:00
|
|
|
return FTC_Err_Invalid_Cache_Handle;
|
2000-08-24 18:29:15 +02:00
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
error = FT_LruList_Lookup( manager->faces_list,
|
|
|
|
(FT_LruKey)face_id,
|
|
|
|
(FT_LruNode*)&node );
|
|
|
|
if ( !error )
|
2001-10-26 18:58:27 +02:00
|
|
|
*aface = node->face;
|
|
|
|
|
|
|
|
return error;
|
2000-08-23 13:22:30 +02:00
|
|
|
}
|
2000-09-19 03:11:11 +02:00
|
|
|
|
|
|
|
|
2001-06-28 19:49:10 +02:00
|
|
|
/* documentation is in ftcache.h */
|
|
|
|
|
2001-06-27 18:18:10 +02:00
|
|
|
FT_EXPORT_DEF( FT_Error )
|
|
|
|
FTC_Manager_Lookup_Size( FTC_Manager manager,
|
|
|
|
FTC_Font font,
|
|
|
|
FT_Face *aface,
|
|
|
|
FT_Size *asize )
|
2000-08-23 13:22:30 +02:00
|
|
|
{
|
2001-10-26 18:58:27 +02:00
|
|
|
FT_Error error;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-08-23 19:32:42 +02:00
|
|
|
|
2000-08-24 18:29:15 +02:00
|
|
|
/* check for valid `manager' delayed to FTC_Manager_Lookup_Face() */
|
|
|
|
if ( aface )
|
2000-08-23 23:11:13 +02:00
|
|
|
*aface = 0;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-08-24 18:29:15 +02:00
|
|
|
if ( asize )
|
2000-08-23 23:11:13 +02:00
|
|
|
*asize = 0;
|
2000-08-23 19:32:42 +02:00
|
|
|
|
2000-10-12 07:05:40 +02:00
|
|
|
error = FTC_Manager_Lookup_Face( manager, font->face_id, aface );
|
2000-08-23 19:32:42 +02:00
|
|
|
if ( !error )
|
2000-08-23 13:22:30 +02:00
|
|
|
{
|
2001-10-26 18:58:27 +02:00
|
|
|
FTC_SizeQueryRec query;
|
|
|
|
FTC_SizeNode node;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-08-24 18:29:15 +02:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
query.face = *aface;
|
|
|
|
query.width = font->pix_width;
|
|
|
|
query.height = font->pix_height;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
error = FT_LruList_Lookup( manager->sizes_list,
|
2001-12-05 02:22:05 +01:00
|
|
|
(FT_LruKey)&query,
|
|
|
|
(FT_LruNode*)&node );
|
2000-08-23 19:32:42 +02:00
|
|
|
if ( !error )
|
2000-08-23 13:22:30 +02:00
|
|
|
{
|
|
|
|
/* select the size as the current one for this face */
|
2001-10-26 18:58:27 +02:00
|
|
|
FT_Activate_Size( node->size );
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-08-24 18:29:15 +02:00
|
|
|
if ( asize )
|
2001-10-26 18:58:27 +02:00
|
|
|
*asize = node->size;
|
2000-08-23 13:22:30 +02:00
|
|
|
}
|
|
|
|
}
|
2000-08-23 19:32:42 +02:00
|
|
|
|
2000-08-23 13:22:30 +02:00
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
/* add a new node to the head of the manager's circular MRU list */
|
2001-10-26 18:58:27 +02:00
|
|
|
static void
|
|
|
|
ftc_node_mru_link( FTC_Node node,
|
|
|
|
FTC_Manager manager )
|
|
|
|
{
|
|
|
|
FTC_Node first = manager->nodes_list;
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
|
|
|
|
if ( first )
|
2001-10-26 18:58:27 +02:00
|
|
|
{
|
2001-12-05 02:22:05 +01:00
|
|
|
node->mru_prev = first->mru_prev;
|
|
|
|
node->mru_next = first;
|
2001-10-26 18:58:27 +02:00
|
|
|
|
|
|
|
first->mru_prev->mru_next = node;
|
|
|
|
first->mru_prev = node;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
node->mru_next = node;
|
|
|
|
node->mru_prev = node;
|
|
|
|
}
|
2001-12-05 02:22:05 +01:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
manager->nodes_list = node;
|
|
|
|
manager->num_nodes++;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
/* remove a node from the manager's MRU list */
|
2001-10-26 18:58:27 +02:00
|
|
|
static void
|
|
|
|
ftc_node_mru_unlink( FTC_Node node,
|
|
|
|
FTC_Manager manager )
|
|
|
|
{
|
|
|
|
FTC_Node prev = node->mru_prev;
|
|
|
|
FTC_Node next = node->mru_next;
|
|
|
|
FTC_Node first = manager->nodes_list;
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
prev->mru_next = next;
|
|
|
|
next->mru_prev = prev;
|
|
|
|
|
|
|
|
if ( node->mru_next == first )
|
|
|
|
{
|
2001-12-05 02:22:05 +01:00
|
|
|
/* this is the last node in the list; update its head pointer */
|
2001-10-26 18:58:27 +02:00
|
|
|
if ( node == first )
|
|
|
|
manager->nodes_list = NULL;
|
|
|
|
else
|
|
|
|
first->mru_prev = prev;
|
|
|
|
}
|
|
|
|
|
|
|
|
node->mru_next = NULL;
|
|
|
|
node->mru_prev = NULL;
|
|
|
|
manager->num_nodes--;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
/* move a node to the head of the manager's MRU list */
|
2001-10-26 18:58:27 +02:00
|
|
|
static void
|
|
|
|
ftc_node_mru_up( FTC_Node node,
|
|
|
|
FTC_Manager manager )
|
|
|
|
{
|
|
|
|
FTC_Node first = manager->nodes_list;
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
if ( node != first )
|
|
|
|
{
|
|
|
|
ftc_node_mru_unlink( node, manager );
|
|
|
|
ftc_node_mru_link( node, manager );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
/* remove a node from its cache's hash table */
|
2001-10-26 18:58:27 +02:00
|
|
|
static void
|
|
|
|
ftc_node_hash_unlink( FTC_Node node,
|
|
|
|
FTC_Cache cache )
|
|
|
|
{
|
2001-12-05 02:22:05 +01:00
|
|
|
FTC_Node *pnode = cache->buckets + ( node->hash % cache->size );
|
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
|
|
|
|
for (;;)
|
|
|
|
{
|
|
|
|
if ( *pnode == NULL )
|
|
|
|
{
|
2001-12-05 02:22:05 +01:00
|
|
|
FT_ERROR(( "FreeType.cache.hash_unlink: unknown node!\n" ));
|
2001-10-26 18:58:27 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( *pnode == node )
|
|
|
|
{
|
|
|
|
*pnode = node->link;
|
|
|
|
node->link = NULL;
|
|
|
|
|
|
|
|
cache->nodes--;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
pnode = &(*pnode)->link;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
/* add a node to the "top" of its cache's hash table */
|
2001-10-26 18:58:27 +02:00
|
|
|
static void
|
|
|
|
ftc_node_hash_link( FTC_Node node,
|
|
|
|
FTC_Cache cache )
|
|
|
|
{
|
2001-12-05 02:22:05 +01:00
|
|
|
FTC_Node *pnode = cache->buckets + ( node->hash % cache->size );
|
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
|
|
|
|
node->link = *pnode;
|
|
|
|
*pnode = node;
|
|
|
|
|
|
|
|
cache->nodes++;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
/* remove a node from the cache manager */
|
2001-10-26 18:58:27 +02:00
|
|
|
static void
|
|
|
|
ftc_node_destroy( FTC_Node node,
|
|
|
|
FTC_Manager manager )
|
|
|
|
{
|
|
|
|
FT_Memory memory = manager->library->memory;
|
|
|
|
FTC_Cache cache;
|
|
|
|
FTC_Cache_Class clazz;
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
#ifdef FT_DEBUG_ERROR
|
|
|
|
/* find node's cache */
|
|
|
|
if ( node->cache_index >= FTC_MAX_CACHES )
|
|
|
|
{
|
|
|
|
FT_ERROR(( "FreeType.cache.node_destroy: invalid node handle\n" ));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
cache = manager->caches[node->cache_index];
|
2001-10-26 18:58:27 +02:00
|
|
|
|
|
|
|
#ifdef FT_DEBUG_ERROR
|
|
|
|
if ( cache == NULL )
|
|
|
|
{
|
|
|
|
FT_ERROR(( "FreeType.cache.node_destroy: invalid node handle\n" ));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
clazz = cache->clazz;
|
|
|
|
|
|
|
|
manager->cur_weight -= clazz->node_weight( node, cache );
|
|
|
|
|
|
|
|
/* remove node from mru list */
|
|
|
|
ftc_node_mru_unlink( node, manager );
|
|
|
|
|
|
|
|
/* remove node from cache's hash table */
|
|
|
|
ftc_node_hash_unlink( node, cache );
|
|
|
|
|
|
|
|
/* now finalize it */
|
|
|
|
|
|
|
|
if ( clazz->node_done )
|
|
|
|
clazz->node_done( node, cache );
|
|
|
|
|
|
|
|
FREE( node );
|
|
|
|
|
|
|
|
/* check, just in case of general corruption :-) */
|
|
|
|
if ( manager->num_nodes <= 0 )
|
|
|
|
FT_ERROR(( "FTC_Manager_Compress: Invalid cache node count! = %d\n",
|
|
|
|
manager->num_nodes ));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
FT_EXPORT_DEF( void )
|
2001-10-26 18:58:27 +02:00
|
|
|
FTC_Manager_Check( FTC_Manager manager )
|
|
|
|
{
|
|
|
|
FTC_Node node, first;
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
first = manager->nodes_list;
|
|
|
|
|
|
|
|
/* check node weights */
|
2001-12-05 02:22:05 +01:00
|
|
|
if ( first )
|
2001-10-26 18:58:27 +02:00
|
|
|
{
|
|
|
|
FT_ULong weight = 0;
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
node = first;
|
2001-12-05 02:22:05 +01:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
do
|
|
|
|
{
|
|
|
|
FTC_Cache cache = manager->caches[node->cache_index];
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
weight += cache->clazz->node_weight( node, cache );
|
2001-12-05 02:22:05 +01:00
|
|
|
node = node->mru_next;
|
|
|
|
|
|
|
|
} while ( node != first );
|
2001-10-26 18:58:27 +02:00
|
|
|
|
|
|
|
if ( weight != manager->cur_weight )
|
2001-12-05 02:22:05 +01:00
|
|
|
FT_ERROR((
|
|
|
|
"FTC_Manager_Compress: invalid weight %ld instead of %ld\n",
|
|
|
|
manager->cur_weight, weight ));
|
2001-10-26 18:58:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* check circular list */
|
2001-12-05 02:22:05 +01:00
|
|
|
if ( first )
|
2001-10-26 18:58:27 +02:00
|
|
|
{
|
|
|
|
FT_UFast count = 0;
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
node = first;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
count++;
|
|
|
|
node = node->mru_next;
|
2001-12-05 02:22:05 +01:00
|
|
|
|
|
|
|
} while ( node != first );
|
2001-10-26 18:58:27 +02:00
|
|
|
|
|
|
|
if ( count != manager->num_nodes )
|
2001-12-05 02:22:05 +01:00
|
|
|
FT_ERROR((
|
|
|
|
"FTC_Manager_Compress: invalid cache node count %d instead of %d\n",
|
|
|
|
manager->num_nodes, count ));
|
2001-10-26 18:58:27 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-10-12 07:05:40 +02:00
|
|
|
/* `Compress' the manager's data, i.e., get rid of old cache nodes */
|
2000-09-19 03:11:11 +02:00
|
|
|
/* that are not referenced anymore in order to limit the total */
|
2000-10-12 07:05:40 +02:00
|
|
|
/* memory used by the cache. */
|
2001-06-28 19:49:10 +02:00
|
|
|
|
|
|
|
/* documentation is in ftcmanag.h */
|
|
|
|
|
2001-06-27 18:18:10 +02:00
|
|
|
FT_EXPORT_DEF( void )
|
|
|
|
FTC_Manager_Compress( FTC_Manager manager )
|
2000-09-19 03:11:11 +02:00
|
|
|
{
|
2001-10-26 18:58:27 +02:00
|
|
|
FTC_Node node, first;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
if ( !manager )
|
|
|
|
return;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
first = manager->nodes_list;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
#if 0
|
|
|
|
FTC_Manager_Check( manager );
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
FT_ERROR(( "compressing, weight = %ld, max = %ld, nodes = %d\n",
|
2001-12-05 02:22:05 +01:00
|
|
|
manager->cur_weight, manager->max_weight,
|
|
|
|
manager->num_nodes ));
|
2001-10-26 18:58:27 +02:00
|
|
|
#endif
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
if ( manager->cur_weight < manager->max_weight || first == NULL )
|
|
|
|
return;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
/* go to last node - it's a circular list */
|
|
|
|
node = first->mru_prev;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
FTC_Node prev = node->mru_prev;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
prev = ( node == first ) ? NULL : node->mru_prev;
|
|
|
|
|
|
|
|
if ( node->ref_count <= 0 )
|
|
|
|
ftc_node_destroy( node, manager );
|
2000-10-31 21:42:18 +01:00
|
|
|
|
2000-09-19 03:11:11 +02:00
|
|
|
node = prev;
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
} while ( node && manager->cur_weight > manager->max_weight );
|
|
|
|
}
|
2000-09-19 03:11:11 +02:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
|
2001-06-27 18:18:10 +02:00
|
|
|
FT_EXPORT_DEF( FT_Error )
|
2001-10-26 18:58:27 +02:00
|
|
|
FTC_Manager_Register_Cache( FTC_Manager manager,
|
|
|
|
FTC_Cache_Class clazz,
|
|
|
|
FTC_Cache *acache )
|
2000-09-19 03:11:11 +02:00
|
|
|
{
|
2001-10-26 18:58:27 +02:00
|
|
|
FT_Error error = FTC_Err_Invalid_Argument;
|
|
|
|
FTC_Cache cache = NULL;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
|
|
|
|
2000-09-19 03:11:11 +02:00
|
|
|
if ( manager && clazz && acache )
|
|
|
|
{
|
|
|
|
FT_Memory memory = manager->library->memory;
|
2001-10-26 18:58:27 +02:00
|
|
|
FT_UInt index = 0;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-09-19 03:11:11 +02:00
|
|
|
|
|
|
|
/* check for an empty cache slot in the manager's table */
|
|
|
|
for ( index = 0; index < FTC_MAX_CACHES; index++ )
|
|
|
|
{
|
|
|
|
if ( manager->caches[index] == 0 )
|
|
|
|
break;
|
|
|
|
}
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2000-09-19 03:11:11 +02:00
|
|
|
/* return an error if there are too many registered caches */
|
2000-10-12 07:05:40 +02:00
|
|
|
if ( index >= FTC_MAX_CACHES )
|
2000-09-19 03:11:11 +02:00
|
|
|
{
|
Complete redesign of error codes. Please check ftmoderr.h for more
details.
* include/freetype/internal/cfferrs.h,
include/freetype/internal/tterrors.h,
include/freetype/internal/t1errors.h: Removed. Replaced with files
local to the module. All extra error codes have been moved to
`fterrors.h'.
* src/sfnt/ttpost.h: Move error codes to `fterrors.h'.
* src/autohint/aherrors.h, src/cache/ftcerror.h, src/cff/cfferrs.h,
src/cid/ciderrs.h, src/pcf/pcferror.h, src/psaux/psauxerr.h,
src/psnames/psnamerr.h, src/raster/rasterrs.h, src/sfnt/sferrors.h,
src/smooth/ftsmerrs.h, src/truetype/tterrors.h,
src/type1/t1errors.h, src/winfonts/fnterrs.h: New files defining the
error names for the module it belongs to.
* include/freetype/ftmoderr.h: New file, defining the module error
offsets. Its structure is similar to `fterrors.h'.
* include/freetype/fterrors.h (FT_NOERRORDEF): New macro.
(FT_ERRORDEF): Redefined to use module error offsets.
All internal error codes are now public; unused error codes have
been removed, some are new.
* include/freetype/config/ftheader.h (FT_MODULE_ERRORS_H): New
macro.
* include/freetype/config/ftoption.h
(FT_CONFIG_OPTION_USE_MODULE_ERRORS): New macro.
All other source files have been updated to use the new error codes;
some already existing (internal) error codes local to a module have
been renamed to give them the same name as in the base module.
All make files have been updated to include the local error files.
* src/cid/cidtokens.h: Replaced with...
* src/cid/cidtoken.h: This file for 8+3 consistency.
* src/raster/ftraster.c: Use macros for header file names.
2001-06-06 19:30:41 +02:00
|
|
|
error = FTC_Err_Too_Many_Caches;
|
2000-09-19 03:11:11 +02:00
|
|
|
FT_ERROR(( "FTC_Manager_Register_Cache:" ));
|
2000-10-12 07:05:40 +02:00
|
|
|
FT_ERROR(( " too many registered caches\n" ));
|
2000-09-19 03:11:11 +02:00
|
|
|
goto Exit;
|
|
|
|
}
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
if ( !ALLOC( cache, clazz->cache_size ) )
|
2000-09-19 03:11:11 +02:00
|
|
|
{
|
|
|
|
cache->manager = manager;
|
|
|
|
cache->memory = memory;
|
|
|
|
cache->clazz = clazz;
|
2000-10-31 21:42:18 +01:00
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
/* THIS IS VERY IMPORTANT! IT WILL WRETCH THE MANAGER */
|
|
|
|
/* IF IT IS NOT SET CORRECTLY */
|
2000-10-31 11:58:23 +01:00
|
|
|
cache->cache_index = index;
|
2000-09-19 03:11:11 +02:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
if ( clazz->cache_init )
|
|
|
|
{
|
|
|
|
error = clazz->cache_init( cache );
|
|
|
|
if ( error )
|
|
|
|
{
|
|
|
|
if ( clazz->cache_done )
|
|
|
|
clazz->cache_done( cache );
|
|
|
|
|
|
|
|
FREE( cache );
|
|
|
|
goto Exit;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
manager->caches[index] = cache;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Exit:
|
|
|
|
*acache = cache;
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*************************************************************************/
|
|
|
|
/*************************************************************************/
|
|
|
|
/***** *****/
|
|
|
|
/***** ABSTRACT CACHE CLASS *****/
|
|
|
|
/***** *****/
|
|
|
|
/*************************************************************************/
|
|
|
|
/*************************************************************************/
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
#define FTC_PRIMES_MIN 7
|
|
|
|
#define FTC_PRIMES_MAX 13845163
|
2001-10-26 18:58:27 +02:00
|
|
|
|
|
|
|
static const FT_UInt ftc_primes[] =
|
|
|
|
{
|
|
|
|
7,
|
|
|
|
11,
|
|
|
|
19,
|
|
|
|
37,
|
|
|
|
73,
|
|
|
|
109,
|
|
|
|
163,
|
|
|
|
251,
|
|
|
|
367,
|
|
|
|
557,
|
|
|
|
823,
|
|
|
|
1237,
|
|
|
|
1861,
|
|
|
|
2777,
|
|
|
|
4177,
|
|
|
|
6247,
|
|
|
|
9371,
|
|
|
|
14057,
|
|
|
|
21089,
|
|
|
|
31627,
|
|
|
|
47431,
|
|
|
|
71143,
|
|
|
|
106721,
|
|
|
|
160073,
|
|
|
|
240101,
|
|
|
|
360163,
|
|
|
|
540217,
|
|
|
|
810343,
|
|
|
|
1215497,
|
|
|
|
1823231,
|
|
|
|
2734867,
|
|
|
|
4102283,
|
|
|
|
6153409,
|
|
|
|
9230113,
|
|
|
|
13845163,
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
static FT_UFast
|
|
|
|
ftc_prime_closest( FT_UFast num )
|
|
|
|
{
|
|
|
|
FT_UInt i;
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
|
|
|
|
for ( i = 0; i < sizeof ( ftc_primes ) / sizeof ( ftc_primes[0] ); i++ )
|
2001-10-26 18:58:27 +02:00
|
|
|
if ( ftc_primes[i] > num )
|
|
|
|
return ftc_primes[i];
|
|
|
|
|
|
|
|
return FTC_PRIMES_MAX;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
FT_EXPORT_DEF( void )
|
|
|
|
ftc_cache_resize( FTC_Cache cache )
|
|
|
|
{
|
|
|
|
FT_UFast new_size;
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
new_size = ftc_prime_closest( cache->nodes );
|
|
|
|
if ( new_size != cache->size )
|
|
|
|
{
|
|
|
|
FT_Memory memory = cache->memory;
|
|
|
|
FT_Error error;
|
|
|
|
FTC_Node* new_buckets ;
|
|
|
|
FT_ULong i;
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
if ( ALLOC_ARRAY( new_buckets, new_size, FTC_Node ) )
|
|
|
|
return;
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
for ( i = 0; i < cache->size; i++ )
|
|
|
|
{
|
|
|
|
FTC_Node node, next, *pnode;
|
|
|
|
FT_UFast hash;
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
node = cache->buckets[i];
|
2001-12-05 02:22:05 +01:00
|
|
|
while ( node )
|
2001-10-26 18:58:27 +02:00
|
|
|
{
|
|
|
|
next = node->link;
|
|
|
|
hash = node->hash % new_size;
|
|
|
|
pnode = new_buckets + hash;
|
|
|
|
|
|
|
|
node->link = pnode[0];
|
|
|
|
pnode[0] = node;
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
node = next;
|
2001-10-26 18:58:27 +02:00
|
|
|
}
|
2000-09-19 03:11:11 +02:00
|
|
|
}
|
2001-10-26 18:58:27 +02:00
|
|
|
|
|
|
|
if ( cache->buckets )
|
|
|
|
FREE( cache->buckets );
|
|
|
|
|
|
|
|
cache->buckets = new_buckets;
|
|
|
|
cache->size = new_size;
|
2000-09-19 03:11:11 +02:00
|
|
|
}
|
2001-10-26 18:58:27 +02:00
|
|
|
}
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
|
|
|
|
FT_EXPORT_DEF( FT_Error )
|
|
|
|
ftc_cache_init( FTC_Cache cache )
|
|
|
|
{
|
|
|
|
FT_Memory memory = cache->memory;
|
|
|
|
FT_Error error;
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
cache->nodes = 0;
|
|
|
|
cache->size = FTC_PRIMES_MIN;
|
|
|
|
|
|
|
|
if ( ALLOC_ARRAY( cache->buckets, cache->size, FTC_Node ) )
|
|
|
|
goto Exit;
|
|
|
|
|
2000-09-19 03:11:11 +02:00
|
|
|
Exit:
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
2000-10-12 07:05:40 +02:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
FT_EXPORT_DEF( void )
|
|
|
|
ftc_cache_done( FTC_Cache cache )
|
|
|
|
{
|
|
|
|
if ( cache )
|
|
|
|
{
|
|
|
|
FT_Memory memory = cache->memory;
|
|
|
|
FTC_Cache_Class clazz = cache->clazz;
|
|
|
|
FTC_Manager manager = cache->manager;
|
|
|
|
FT_UFast i;
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
for ( i = 0; i < cache->size; i++ )
|
|
|
|
{
|
|
|
|
FTC_Node *pnode = cache->buckets + i, next, node = *pnode;
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
while ( node )
|
2001-10-26 18:58:27 +02:00
|
|
|
{
|
|
|
|
next = node->link;
|
|
|
|
node->link = NULL;
|
|
|
|
|
|
|
|
/* remove node from mru list */
|
|
|
|
ftc_node_mru_unlink( node, manager );
|
|
|
|
|
|
|
|
/* now finalize it */
|
|
|
|
manager->cur_weight -= clazz->node_weight( node, cache );
|
|
|
|
|
|
|
|
if ( clazz->node_done )
|
|
|
|
clazz->node_done( node, cache );
|
|
|
|
|
|
|
|
FREE( node );
|
|
|
|
node = next;
|
|
|
|
}
|
|
|
|
cache->buckets[i] = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
FREE( cache->buckets );
|
|
|
|
|
|
|
|
cache->nodes = 0;
|
|
|
|
cache->size = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
/* Look up a node in "top" of its cache's hash table. */
|
|
|
|
/* If not found, create a new node. */
|
|
|
|
/* */
|
|
|
|
FT_EXPORT_DEF( FT_Error )
|
2001-10-26 18:58:27 +02:00
|
|
|
ftc_cache_lookup_node( FTC_Cache cache,
|
|
|
|
FT_UFast key_hash,
|
|
|
|
FT_Pointer key,
|
|
|
|
FTC_Node *anode )
|
|
|
|
{
|
|
|
|
FT_Error error = 0;
|
|
|
|
FTC_Node result = NULL;
|
2001-12-05 02:22:05 +01:00
|
|
|
FTC_Node* bucket = cache->buckets + ( key_hash % cache->size );
|
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
|
|
|
|
if ( *bucket )
|
|
|
|
{
|
|
|
|
FTC_Node* pnode = bucket;
|
|
|
|
FTC_Node_CompareFunc compare = cache->clazz->node_compare;
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
|
|
|
|
for (;;)
|
2001-10-26 18:58:27 +02:00
|
|
|
{
|
|
|
|
FTC_Node node;
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
node = *pnode;
|
|
|
|
if ( node == NULL )
|
|
|
|
break;
|
|
|
|
|
|
|
|
if ( compare( node, key, cache ) )
|
|
|
|
{
|
|
|
|
/* move to head of bucket list */
|
|
|
|
if ( pnode != bucket )
|
|
|
|
{
|
|
|
|
*pnode = node->link;
|
|
|
|
node->link = *bucket;
|
|
|
|
*bucket = node;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* move to head of MRU list */
|
|
|
|
if ( node != cache->manager->nodes_list )
|
|
|
|
ftc_node_mru_up( node, cache->manager );
|
|
|
|
|
|
|
|
result = node;
|
|
|
|
goto Exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
pnode = &(*pnode)->link;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* didn't find a node, create a new one */
|
|
|
|
{
|
|
|
|
FTC_Cache_Class clazz = cache->clazz;
|
|
|
|
FTC_Manager manager = cache->manager;
|
|
|
|
FT_Memory memory = cache->memory;
|
|
|
|
FTC_Node node;
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
|
2001-10-26 18:58:27 +02:00
|
|
|
if ( ALLOC( node, clazz->node_size ) )
|
|
|
|
goto Exit;
|
|
|
|
|
|
|
|
/* node initializer must set 'hash' field */
|
|
|
|
|
|
|
|
node->cache_index = cache->cache_index;
|
|
|
|
node->ref_count = 0;
|
|
|
|
|
|
|
|
error = clazz->node_init( node, key, cache );
|
|
|
|
if ( error )
|
|
|
|
{
|
|
|
|
FREE( node );
|
|
|
|
goto Exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
ftc_node_hash_link( node, cache );
|
|
|
|
ftc_node_mru_link( node, cache->manager );
|
|
|
|
|
|
|
|
cache->manager->cur_weight += clazz->node_weight( node, cache );
|
|
|
|
|
|
|
|
/* now try to compress the node pool when necessary */
|
|
|
|
if ( manager->cur_weight >= manager->max_weight )
|
|
|
|
{
|
|
|
|
node->ref_count++;
|
|
|
|
FTC_Manager_Compress( manager );
|
|
|
|
node->ref_count--;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* try to resize the hash table when appropriate */
|
2001-12-05 02:22:05 +01:00
|
|
|
if ( FTC_CACHE_RESIZE_TEST( cache ) )
|
2001-10-26 18:58:27 +02:00
|
|
|
ftc_cache_resize( cache );
|
|
|
|
|
|
|
|
result = node;
|
|
|
|
}
|
|
|
|
|
|
|
|
Exit:
|
|
|
|
*anode = result;
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
/* maybe the next two functions will disappear eventually */
|
2001-10-26 18:58:27 +02:00
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
FT_EXPORT_DEF( void )
|
2001-10-26 18:58:27 +02:00
|
|
|
ftc_node_ref( FTC_Node node,
|
|
|
|
FTC_Cache cache )
|
|
|
|
{
|
|
|
|
if ( node && cache && (FT_UInt)node->cache_index == cache->cache_index )
|
|
|
|
node->ref_count++;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2001-12-05 02:22:05 +01:00
|
|
|
FT_EXPORT_DEF( void )
|
2001-10-26 18:58:27 +02:00
|
|
|
ftc_node_unref( FTC_Node node,
|
|
|
|
FTC_Cache cache )
|
|
|
|
{
|
|
|
|
if ( node && cache && (FT_UInt)node->cache_index == cache->cache_index )
|
|
|
|
node->ref_count--;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-08-23 19:32:42 +02:00
|
|
|
/* END */
|