2018-06-03 09:01:17 +02:00
|
|
|
/****************************************************************************
|
|
|
|
*
|
|
|
|
* ttbdf.c
|
|
|
|
*
|
|
|
|
* TrueType and OpenType embedded BDF properties (body).
|
|
|
|
*
|
2020-01-19 17:05:19 +01:00
|
|
|
* Copyright (C) 2005-2020 by
|
2018-06-03 09:01:17 +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.
|
|
|
|
*
|
|
|
|
*/
|
2006-02-25 17:52:16 +01:00
|
|
|
|
|
|
|
|
2020-06-08 13:31:55 +02:00
|
|
|
#include <freetype/internal/ftdebug.h>
|
|
|
|
#include <freetype/internal/ftstream.h>
|
|
|
|
#include <freetype/tttags.h>
|
2006-02-25 17:52:16 +01:00
|
|
|
#include "ttbdf.h"
|
|
|
|
|
|
|
|
#include "sferrors.h"
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef TT_CONFIG_OPTION_BDF
|
|
|
|
|
2018-06-03 09:01:17 +02:00
|
|
|
/**************************************************************************
|
|
|
|
*
|
|
|
|
* The macro FT_COMPONENT is used in trace mode. It is an implicit
|
|
|
|
* parameter of the FT_TRACE() and FT_ERROR() macros, used to print/log
|
|
|
|
* messages during execution.
|
|
|
|
*/
|
2006-02-25 17:52:16 +01:00
|
|
|
#undef FT_COMPONENT
|
2018-08-15 18:13:17 +02:00
|
|
|
#define FT_COMPONENT ttbdf
|
2006-02-25 17:52:16 +01:00
|
|
|
|
|
|
|
|
|
|
|
FT_LOCAL_DEF( void )
|
|
|
|
tt_face_free_bdf_props( TT_Face face )
|
|
|
|
{
|
|
|
|
TT_BDF bdf = &face->bdf;
|
|
|
|
|
|
|
|
|
|
|
|
if ( bdf->loaded )
|
|
|
|
{
|
2018-06-16 21:30:04 +02:00
|
|
|
FT_Stream stream = FT_FACE( face )->stream;
|
2006-02-25 17:52:16 +01:00
|
|
|
|
|
|
|
|
2016-12-26 17:08:17 +01:00
|
|
|
if ( bdf->table )
|
2006-02-25 17:52:16 +01:00
|
|
|
FT_FRAME_RELEASE( bdf->table );
|
|
|
|
|
|
|
|
bdf->table_end = NULL;
|
|
|
|
bdf->strings = NULL;
|
|
|
|
bdf->strings_size = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static FT_Error
|
|
|
|
tt_face_load_bdf_props( TT_Face face,
|
|
|
|
FT_Stream stream )
|
|
|
|
{
|
|
|
|
TT_BDF bdf = &face->bdf;
|
|
|
|
FT_ULong length;
|
|
|
|
FT_Error error;
|
|
|
|
|
|
|
|
|
|
|
|
FT_ZERO( bdf );
|
|
|
|
|
|
|
|
error = tt_face_goto_table( face, TTAG_BDF, stream, &length );
|
|
|
|
if ( error ||
|
|
|
|
length < 8 ||
|
|
|
|
FT_FRAME_EXTRACT( length, bdf->table ) )
|
|
|
|
{
|
2013-03-14 10:27:35 +01:00
|
|
|
error = FT_THROW( Invalid_Table );
|
2006-02-25 17:52:16 +01:00
|
|
|
goto Exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
bdf->table_end = bdf->table + length;
|
|
|
|
|
|
|
|
{
|
|
|
|
FT_Byte* p = bdf->table;
|
|
|
|
FT_UInt version = FT_NEXT_USHORT( p );
|
|
|
|
FT_UInt num_strikes = FT_NEXT_USHORT( p );
|
2009-07-31 17:32:13 +02:00
|
|
|
FT_ULong strings = FT_NEXT_ULONG ( p );
|
2006-02-25 17:52:16 +01:00
|
|
|
FT_UInt count;
|
|
|
|
FT_Byte* strike;
|
|
|
|
|
|
|
|
|
|
|
|
if ( version != 0x0001 ||
|
|
|
|
strings < 8 ||
|
|
|
|
( strings - 8 ) / 4 < num_strikes ||
|
|
|
|
strings + 1 > length )
|
|
|
|
{
|
|
|
|
goto BadTable;
|
|
|
|
}
|
|
|
|
|
|
|
|
bdf->num_strikes = num_strikes;
|
|
|
|
bdf->strings = bdf->table + strings;
|
|
|
|
bdf->strings_size = length - strings;
|
|
|
|
|
|
|
|
count = bdf->num_strikes;
|
|
|
|
p = bdf->table + 8;
|
|
|
|
strike = p + count * 4;
|
|
|
|
|
|
|
|
|
|
|
|
for ( ; count > 0; count-- )
|
|
|
|
{
|
|
|
|
FT_UInt num_items = FT_PEEK_USHORT( p + 2 );
|
|
|
|
|
|
|
|
/*
|
2018-06-03 09:01:17 +02:00
|
|
|
* We don't need to check the value sets themselves, since this
|
|
|
|
* is done later.
|
2006-02-25 17:52:16 +01:00
|
|
|
*/
|
|
|
|
strike += 10 * num_items;
|
|
|
|
|
|
|
|
p += 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( strike > bdf->strings )
|
|
|
|
goto BadTable;
|
|
|
|
}
|
|
|
|
|
|
|
|
bdf->loaded = 1;
|
|
|
|
|
|
|
|
Exit:
|
|
|
|
return error;
|
|
|
|
|
|
|
|
BadTable:
|
|
|
|
FT_FRAME_RELEASE( bdf->table );
|
|
|
|
FT_ZERO( bdf );
|
2013-03-14 10:27:35 +01:00
|
|
|
error = FT_THROW( Invalid_Table );
|
2006-02-25 17:52:16 +01:00
|
|
|
goto Exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
FT_LOCAL_DEF( FT_Error )
|
|
|
|
tt_face_find_bdf_prop( TT_Face face,
|
|
|
|
const char* property_name,
|
|
|
|
BDF_PropertyRec *aprop )
|
|
|
|
{
|
2009-07-31 17:30:14 +02:00
|
|
|
TT_BDF bdf = &face->bdf;
|
2018-06-16 21:30:04 +02:00
|
|
|
FT_Size size = FT_FACE( face )->size;
|
2013-03-14 11:21:17 +01:00
|
|
|
FT_Error error = FT_Err_Ok;
|
2009-07-31 17:30:14 +02:00
|
|
|
FT_Byte* p;
|
|
|
|
FT_UInt count;
|
|
|
|
FT_Byte* strike;
|
|
|
|
FT_Offset property_len;
|
2006-02-25 17:52:16 +01:00
|
|
|
|
|
|
|
|
|
|
|
aprop->type = BDF_PROPERTY_TYPE_NONE;
|
|
|
|
|
|
|
|
if ( bdf->loaded == 0 )
|
|
|
|
{
|
|
|
|
error = tt_face_load_bdf_props( face, FT_FACE( face )->stream );
|
|
|
|
if ( error )
|
|
|
|
goto Exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
count = bdf->num_strikes;
|
|
|
|
p = bdf->table + 8;
|
|
|
|
strike = p + 4 * count;
|
|
|
|
|
2013-03-14 17:50:49 +01:00
|
|
|
error = FT_ERR( Invalid_Argument );
|
2006-02-25 17:52:16 +01:00
|
|
|
|
2016-12-26 17:08:17 +01:00
|
|
|
if ( !size || !property_name )
|
2006-02-25 17:52:16 +01:00
|
|
|
goto Exit;
|
|
|
|
|
|
|
|
property_len = ft_strlen( property_name );
|
|
|
|
if ( property_len == 0 )
|
|
|
|
goto Exit;
|
|
|
|
|
|
|
|
for ( ; count > 0; count-- )
|
|
|
|
{
|
|
|
|
FT_UInt _ppem = FT_NEXT_USHORT( p );
|
|
|
|
FT_UInt _count = FT_NEXT_USHORT( p );
|
|
|
|
|
2016-12-26 17:08:17 +01:00
|
|
|
|
2006-02-25 17:52:16 +01:00
|
|
|
if ( _ppem == size->metrics.y_ppem )
|
|
|
|
{
|
|
|
|
count = _count;
|
|
|
|
goto FoundStrike;
|
|
|
|
}
|
|
|
|
|
|
|
|
strike += 10 * _count;
|
|
|
|
}
|
|
|
|
goto Exit;
|
|
|
|
|
|
|
|
FoundStrike:
|
|
|
|
p = strike;
|
|
|
|
for ( ; count > 0; count-- )
|
|
|
|
{
|
|
|
|
FT_UInt type = FT_PEEK_USHORT( p + 4 );
|
|
|
|
|
2016-12-26 17:08:17 +01:00
|
|
|
|
2006-02-25 17:52:16 +01:00
|
|
|
if ( ( type & 0x10 ) != 0 )
|
|
|
|
{
|
|
|
|
FT_UInt32 name_offset = FT_PEEK_ULONG( p );
|
|
|
|
FT_UInt32 value = FT_PEEK_ULONG( p + 6 );
|
|
|
|
|
|
|
|
/* be a bit paranoid for invalid entries here */
|
|
|
|
if ( name_offset < bdf->strings_size &&
|
|
|
|
property_len < bdf->strings_size - name_offset &&
|
|
|
|
ft_strncmp( property_name,
|
|
|
|
(const char*)bdf->strings + name_offset,
|
|
|
|
bdf->strings_size - name_offset ) == 0 )
|
|
|
|
{
|
|
|
|
switch ( type & 0x0F )
|
|
|
|
{
|
|
|
|
case 0x00: /* string */
|
|
|
|
case 0x01: /* atoms */
|
|
|
|
/* check that the content is really 0-terminated */
|
|
|
|
if ( value < bdf->strings_size &&
|
|
|
|
ft_memchr( bdf->strings + value, 0, bdf->strings_size ) )
|
|
|
|
{
|
|
|
|
aprop->type = BDF_PROPERTY_TYPE_ATOM;
|
|
|
|
aprop->u.atom = (const char*)bdf->strings + value;
|
2013-03-14 11:21:17 +01:00
|
|
|
error = FT_Err_Ok;
|
2006-02-25 17:52:16 +01:00
|
|
|
goto Exit;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x02:
|
|
|
|
aprop->type = BDF_PROPERTY_TYPE_INTEGER;
|
|
|
|
aprop->u.integer = (FT_Int32)value;
|
2013-03-14 11:21:17 +01:00
|
|
|
error = FT_Err_Ok;
|
2006-02-25 17:52:16 +01:00
|
|
|
goto Exit;
|
|
|
|
|
|
|
|
case 0x03:
|
|
|
|
aprop->type = BDF_PROPERTY_TYPE_CARDINAL;
|
|
|
|
aprop->u.cardinal = value;
|
2013-03-14 11:21:17 +01:00
|
|
|
error = FT_Err_Ok;
|
2006-02-25 17:52:16 +01:00
|
|
|
goto Exit;
|
|
|
|
|
|
|
|
default:
|
|
|
|
;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
p += 10;
|
|
|
|
}
|
|
|
|
|
|
|
|
Exit:
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
Improve `make multi'.
* src/autofit/aflatin2.c: Guard file with FT_OPTION_AUTOFIT2.
* src/base/ftmac.c: Guard more parts of the file with FT_MACINTOSH.
* src/psaux/afmparse.c: Guard file with T1_CONFIG_OPTION_NO_AFM.
* src/sfnt/pngshim.c: Guard file with
TT_CONFIG_OPTION_EMBEDDED_BITMAPS also.
* src/sfnt/ttbdf.c: Avoid empty source file.
* src/sfnt/ttpost.c: Guard file with
TT_CONFIG_OPTION_POSTSCRIPT_NAMES.
* src/sfnt/ttsbit.c: Guard file with
TT_CONFIG_OPTION_EMBEDDED_BITMAPS.
* src/truetype/ttgxvar.c, src/truetype/ttinterp.c: Avoid empty
source file.
* src/truetype/ttsubpix.c: Guard file with
TT_USE_BYTECODE_INTERPRETER also.
* src/type1/t1afm.c: Guard file with T1_CONFIG_OPTION_NO_AFM.
* src/autofit/autofit.c, src/base/ftbase.c, src/cache/ftcache.c,
src/cff/cff.c, src/cid/type1cid.c, src/gxvalid/gxvalid.c,
src/pcf/pcf.c, src/pfr/pfr.c, src/psaux/psaux.c,
src/pshinter/pshinter.c, src/psnames/psnames.c, src/raster/raster.c,
src/sfnt/sfnt.c, src/smooth/smooth.c, src/truetype/truetype.c,
src/type1/type1.c, src/type42/type42.c: Remove conditionals; sort
entries.
2017-03-18 07:06:49 +01:00
|
|
|
#else /* !TT_CONFIG_OPTION_BDF */
|
|
|
|
|
|
|
|
/* ANSI C doesn't like empty source files */
|
|
|
|
typedef int _tt_bdf_dummy;
|
|
|
|
|
|
|
|
#endif /* !TT_CONFIG_OPTION_BDF */
|
2006-02-25 17:52:16 +01:00
|
|
|
|
|
|
|
|
|
|
|
/* END */
|