630 lines
16 KiB
C
630 lines
16 KiB
C
|
#include "pfrsbit.h"
|
||
|
#include "pfrload.h"
|
||
|
#include FT_INTERNAL_DEBUG_H
|
||
|
#include FT_INTERNAL_STREAM_H
|
||
|
|
||
|
#include "pfrerror.h"
|
||
|
|
||
|
#undef FT_COMPONENT
|
||
|
#define FT_COMPONENT trace_pfr
|
||
|
|
||
|
/*************************************************************************/
|
||
|
/*************************************************************************/
|
||
|
/***** *****/
|
||
|
/***** PFR BIT WRITER *****/
|
||
|
/***** *****/
|
||
|
/*************************************************************************/
|
||
|
/*************************************************************************/
|
||
|
|
||
|
typedef struct PFR_BitWriter_
|
||
|
{
|
||
|
FT_Byte* line; /* current line start */
|
||
|
FT_Int pitch; /* line size in bytes */
|
||
|
FT_Int width; /* width in pixels/bits */
|
||
|
FT_Int rows; /* number of remaining rows to scan */
|
||
|
FT_Int total; /* total number of bits to draw */
|
||
|
|
||
|
} PFR_BitWriterRec, *PFR_BitWriter;
|
||
|
|
||
|
|
||
|
static void
|
||
|
pfr_bitwriter_init( PFR_BitWriter writer,
|
||
|
FT_Bitmap* target,
|
||
|
FT_Bool decreasing )
|
||
|
{
|
||
|
writer->line = target->buffer;
|
||
|
writer->pitch = target->pitch;
|
||
|
writer->width = target->width;
|
||
|
writer->rows = target->rows;
|
||
|
writer->total = writer->width * writer->rows;
|
||
|
|
||
|
if ( !decreasing )
|
||
|
{
|
||
|
writer->line += writer->pitch * ( target->rows-1 );
|
||
|
writer->pitch = -writer->pitch;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
static void
|
||
|
pfr_bitwriter_decode_bytes( PFR_BitWriter writer,
|
||
|
FT_Byte* p,
|
||
|
FT_Byte* limit )
|
||
|
{
|
||
|
FT_Int n, reload;
|
||
|
FT_Int left = writer->width;
|
||
|
FT_Byte* cur = writer->line;
|
||
|
FT_UInt mask = 0x80;
|
||
|
FT_UInt val = 0;
|
||
|
FT_UInt c = 0;
|
||
|
|
||
|
n = (FT_Int)(limit - p)*8;
|
||
|
if ( n > writer->total )
|
||
|
n = writer->total;
|
||
|
|
||
|
reload = n & 7;
|
||
|
|
||
|
for ( ; n > 0; n-- )
|
||
|
{
|
||
|
if ( (n & 7) == reload )
|
||
|
val = *p++;
|
||
|
|
||
|
if ( val & 0x80 )
|
||
|
c |= mask;
|
||
|
|
||
|
val <<= 1;
|
||
|
mask >>= 1;
|
||
|
|
||
|
if ( --left <= 0 )
|
||
|
{
|
||
|
cur[0] = (FT_Byte) c;
|
||
|
left = writer->width;
|
||
|
mask = 0x80;
|
||
|
|
||
|
writer->line += writer->pitch;
|
||
|
cur = writer->line;
|
||
|
c = 0;
|
||
|
}
|
||
|
else if ( mask == 0 )
|
||
|
{
|
||
|
cur[0] = c;
|
||
|
mask = 0x80;
|
||
|
c = 0;
|
||
|
cur ++;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( mask != 0x80 )
|
||
|
cur[0] = c;
|
||
|
}
|
||
|
|
||
|
|
||
|
static void
|
||
|
pfr_bitwriter_decode_rle1( PFR_BitWriter writer,
|
||
|
FT_Byte* p,
|
||
|
FT_Byte* limit )
|
||
|
{
|
||
|
FT_Int n, phase, count, counts[2], reload;
|
||
|
FT_Int left = writer->width;
|
||
|
FT_Byte* cur = writer->line;
|
||
|
FT_UInt mask = 0x80;
|
||
|
FT_UInt c = 0;
|
||
|
|
||
|
n = writer->total;
|
||
|
|
||
|
phase = 1;
|
||
|
counts[0] = 0;
|
||
|
counts[1] = 0;
|
||
|
count = 0;
|
||
|
reload = 1;
|
||
|
|
||
|
for ( ; n > 0; n-- )
|
||
|
{
|
||
|
if ( reload )
|
||
|
{
|
||
|
do
|
||
|
{
|
||
|
if ( phase )
|
||
|
{
|
||
|
FT_Int v;
|
||
|
|
||
|
if ( p >= limit )
|
||
|
break;
|
||
|
|
||
|
v = *p++;
|
||
|
counts[0] = (v >> 4);
|
||
|
counts[1] = (v & 15);
|
||
|
phase = 0;
|
||
|
count = counts[0];
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
phase = 1;
|
||
|
count = counts[1];
|
||
|
}
|
||
|
}
|
||
|
while ( count == 0 );
|
||
|
}
|
||
|
|
||
|
if ( phase )
|
||
|
c |= mask;
|
||
|
|
||
|
mask >>= 1;
|
||
|
|
||
|
if ( --left <= 0 )
|
||
|
{
|
||
|
cur[0] = (FT_Byte) c;
|
||
|
left = writer->width;
|
||
|
mask = 0x80;
|
||
|
|
||
|
writer->line += writer->pitch;
|
||
|
cur = writer->line;
|
||
|
c = 0;
|
||
|
}
|
||
|
else if ( mask == 0 )
|
||
|
{
|
||
|
cur[0] = c;
|
||
|
mask = 0x80;
|
||
|
c = 0;
|
||
|
cur ++;
|
||
|
}
|
||
|
|
||
|
reload = ( --count <= 0 );
|
||
|
}
|
||
|
|
||
|
if ( mask != 0x80 )
|
||
|
cur[0] = (FT_Byte) c;
|
||
|
}
|
||
|
|
||
|
|
||
|
static void
|
||
|
pfr_bitwriter_decode_rle2( PFR_BitWriter writer,
|
||
|
FT_Byte* p,
|
||
|
FT_Byte* limit )
|
||
|
{
|
||
|
FT_Int n, phase, count, reload;
|
||
|
FT_Int left = writer->width;
|
||
|
FT_Byte* cur = writer->line;
|
||
|
FT_UInt mask = 0x80;
|
||
|
FT_UInt c = 0;
|
||
|
|
||
|
n = writer->total;
|
||
|
|
||
|
phase = 1;
|
||
|
count = 0;
|
||
|
reload = 1;
|
||
|
|
||
|
for ( ; n > 0; n-- )
|
||
|
{
|
||
|
if ( reload )
|
||
|
{
|
||
|
do
|
||
|
{
|
||
|
if ( p >= limit )
|
||
|
break;
|
||
|
|
||
|
count = *p++;
|
||
|
phase = phase ^ 1;
|
||
|
}
|
||
|
while ( count == 0 );
|
||
|
}
|
||
|
|
||
|
if ( phase )
|
||
|
c |= mask;
|
||
|
|
||
|
mask >>= 1;
|
||
|
|
||
|
if ( --left <= 0 )
|
||
|
{
|
||
|
cur[0] = (FT_Byte) c;
|
||
|
c = 0;
|
||
|
mask = 0x80;
|
||
|
left = writer->width;
|
||
|
|
||
|
writer->line += writer->pitch;
|
||
|
cur = writer->line;
|
||
|
}
|
||
|
else if ( mask == 0 )
|
||
|
{
|
||
|
cur[0] = c;
|
||
|
c = 0;
|
||
|
mask = 0x80;
|
||
|
cur ++;
|
||
|
}
|
||
|
|
||
|
reload = ( --count <= 0 );
|
||
|
}
|
||
|
|
||
|
if ( mask != 0x80 )
|
||
|
cur[0] = (FT_Byte) c;
|
||
|
}
|
||
|
|
||
|
|
||
|
/*************************************************************************/
|
||
|
/*************************************************************************/
|
||
|
/***** *****/
|
||
|
/***** BITMAP DATA DECODING *****/
|
||
|
/***** *****/
|
||
|
/*************************************************************************/
|
||
|
/*************************************************************************/
|
||
|
|
||
|
static void
|
||
|
pfr_lookup_bitmap_data( FT_Byte* base,
|
||
|
FT_Byte* limit,
|
||
|
FT_Int count,
|
||
|
FT_Byte flags,
|
||
|
FT_UInt char_code,
|
||
|
FT_ULong* found_offset,
|
||
|
FT_ULong* found_size )
|
||
|
{
|
||
|
FT_UInt left, right, char_len;
|
||
|
FT_Bool two = (flags & 1);
|
||
|
FT_Byte* buff;
|
||
|
|
||
|
char_len = 4;
|
||
|
if ( two ) char_len += 1;
|
||
|
if ( flags & 2) char_len += 1;
|
||
|
if ( flags & 4) char_len += 1;
|
||
|
|
||
|
left = 0;
|
||
|
right = count;
|
||
|
|
||
|
while ( left < right )
|
||
|
{
|
||
|
FT_UInt middle, code;
|
||
|
|
||
|
middle = (left + right) >> 1;
|
||
|
buff = base + middle*char_len;
|
||
|
|
||
|
/* check that we're not outside of the table */
|
||
|
/* this is possible with broken fonts... */
|
||
|
if ( buff + char_len > limit )
|
||
|
goto Fail;
|
||
|
|
||
|
if (two) code = PFR_NEXT_USHORT(buff);
|
||
|
else code = PFR_NEXT_BYTE(buff);
|
||
|
|
||
|
if ( code == char_code )
|
||
|
goto Found_It;
|
||
|
|
||
|
if ( code < char_code )
|
||
|
left = middle;
|
||
|
else
|
||
|
right = middle;
|
||
|
}
|
||
|
|
||
|
Fail:
|
||
|
/* Not found */
|
||
|
*found_size = 0;
|
||
|
*found_offset = 0;
|
||
|
return;
|
||
|
|
||
|
Found_It:
|
||
|
if (flags & 2) *found_size = PFR_NEXT_USHORT(buff);
|
||
|
else *found_size = PFR_NEXT_BYTE(buff);
|
||
|
|
||
|
if (flags & 4) *found_offset = PFR_NEXT_ULONG(buff);
|
||
|
else *found_offset = PFR_NEXT_USHORT(buff);
|
||
|
}
|
||
|
|
||
|
|
||
|
/* load bitmap metrics. "*padvance" must be set to the default value */
|
||
|
/* before calling this function... */
|
||
|
/* */
|
||
|
static FT_Error
|
||
|
pfr_load_bitmap_metrics( FT_Byte** pdata,
|
||
|
FT_Byte* limit,
|
||
|
FT_Long scaled_advance,
|
||
|
FT_Long *axpos,
|
||
|
FT_Long *aypos,
|
||
|
FT_UInt *axsize,
|
||
|
FT_UInt *aysize,
|
||
|
FT_Long *aadvance,
|
||
|
FT_UInt *aformat )
|
||
|
{
|
||
|
FT_Error error = 0;
|
||
|
FT_Byte flags;
|
||
|
FT_Char b;
|
||
|
FT_Byte* p = *pdata;
|
||
|
FT_Long xpos, ypos, advance;
|
||
|
FT_UInt xsize, ysize;
|
||
|
|
||
|
PFR_CHECK(1);
|
||
|
flags = PFR_NEXT_BYTE(p);
|
||
|
|
||
|
xpos = 0;
|
||
|
ypos = 0;
|
||
|
xsize = 0;
|
||
|
ysize = 0;
|
||
|
advance = 0;
|
||
|
|
||
|
switch (flags & 3)
|
||
|
{
|
||
|
case 0:
|
||
|
PFR_CHECK(1);
|
||
|
b = PFR_NEXT_INT8(p);
|
||
|
xpos = b >> 4;
|
||
|
ypos = ((FT_Char)(b << 4)) >> 4;
|
||
|
break;
|
||
|
|
||
|
case 1:
|
||
|
PFR_CHECK(2);
|
||
|
xpos = PFR_NEXT_INT8(p);
|
||
|
ypos = PFR_NEXT_INT8(p);
|
||
|
break;
|
||
|
|
||
|
case 2:
|
||
|
PFR_CHECK(4);
|
||
|
xpos = PFR_NEXT_SHORT(p);
|
||
|
ypos = PFR_NEXT_SHORT(p);
|
||
|
break;
|
||
|
|
||
|
case 3:
|
||
|
PFR_CHECK(6);
|
||
|
xpos = PFR_NEXT_LONG(p);
|
||
|
ypos = PFR_NEXT_LONG(p);
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
;
|
||
|
}
|
||
|
|
||
|
flags >>= 2;
|
||
|
switch (flags & 3)
|
||
|
{
|
||
|
case 0:
|
||
|
/* blank image */
|
||
|
xsize = 0;
|
||
|
ysize = 0;
|
||
|
break;
|
||
|
|
||
|
case 1:
|
||
|
PFR_CHECK(1);
|
||
|
b = PFR_NEXT_BYTE(p);
|
||
|
xsize = (b >> 4) & 0xF;
|
||
|
ysize = b & 0xF;
|
||
|
break;
|
||
|
|
||
|
case 2:
|
||
|
PFR_CHECK(2);
|
||
|
xsize = PFR_NEXT_BYTE(p);
|
||
|
ysize = PFR_NEXT_BYTE(p);
|
||
|
break;
|
||
|
|
||
|
case 3:
|
||
|
PFR_CHECK(4);
|
||
|
xsize = PFR_NEXT_USHORT(p);
|
||
|
ysize = PFR_NEXT_USHORT(p);
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
;
|
||
|
}
|
||
|
|
||
|
flags >>= 2;
|
||
|
switch (flags & 3)
|
||
|
{
|
||
|
case 0:
|
||
|
advance = scaled_advance;
|
||
|
break;
|
||
|
|
||
|
case 1:
|
||
|
PFR_CHECK(1);
|
||
|
advance = PFR_NEXT_INT8(p) << 8;
|
||
|
break;
|
||
|
|
||
|
case 2:
|
||
|
PFR_CHECK(2);
|
||
|
advance = PFR_NEXT_SHORT(p);
|
||
|
break;
|
||
|
|
||
|
case 3:
|
||
|
PFR_CHECK(3);
|
||
|
advance = PFR_NEXT_LONG(p);
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
;
|
||
|
}
|
||
|
|
||
|
*axpos = xpos;
|
||
|
*aypos = ypos;
|
||
|
*axsize = xsize;
|
||
|
*aysize = ysize;
|
||
|
*aadvance = advance;
|
||
|
*aformat = flags >> 2;
|
||
|
*pdata = p;
|
||
|
|
||
|
Exit:
|
||
|
return error;
|
||
|
|
||
|
Too_Short:
|
||
|
error = PFR_Err_Invalid_Table;
|
||
|
FT_ERROR(( "pfr_load_bitmap_metrics: invalid glyph data\n" ));
|
||
|
goto Exit;
|
||
|
}
|
||
|
|
||
|
|
||
|
static FT_Error
|
||
|
pfr_load_bitmap_bits( FT_Byte* p,
|
||
|
FT_Byte* limit,
|
||
|
FT_UInt format,
|
||
|
FT_UInt decreasing,
|
||
|
FT_Bitmap* target )
|
||
|
{
|
||
|
FT_Error error = 0;
|
||
|
PFR_BitWriterRec writer;
|
||
|
|
||
|
if ( target->rows > 0 && target->width > 0 )
|
||
|
{
|
||
|
pfr_bitwriter_init( &writer, target, decreasing );
|
||
|
|
||
|
switch (format)
|
||
|
{
|
||
|
case 0: /* packed bits */
|
||
|
pfr_bitwriter_decode_bytes( &writer, p, limit );
|
||
|
break;
|
||
|
|
||
|
case 1: /* RLE1 */
|
||
|
pfr_bitwriter_decode_rle1( &writer, p, limit );
|
||
|
break;
|
||
|
|
||
|
case 2: /* RLE2 */
|
||
|
pfr_bitwriter_decode_rle2( &writer, p, limit );
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
FT_ERROR(( "pfr_read_bitmap_data: invalid image type\n" ));
|
||
|
error = FT_Err_Invalid_File_Format;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return error;
|
||
|
}
|
||
|
|
||
|
|
||
|
/*************************************************************************/
|
||
|
/*************************************************************************/
|
||
|
/***** *****/
|
||
|
/***** BITMAP LOADING *****/
|
||
|
/***** *****/
|
||
|
/*************************************************************************/
|
||
|
/*************************************************************************/
|
||
|
|
||
|
FT_LOCAL( FT_Error )
|
||
|
pfr_slot_load_bitmap( PFR_Slot glyph,
|
||
|
PFR_Size size,
|
||
|
FT_UInt glyph_index )
|
||
|
{
|
||
|
FT_Error error;
|
||
|
PFR_Face face = (PFR_Face) glyph->root.face;
|
||
|
FT_Stream stream = face->root.stream;
|
||
|
PFR_PhyFont phys = &face->phy_font;
|
||
|
FT_ULong gps_offset;
|
||
|
FT_ULong gps_size;
|
||
|
PFR_Char character;
|
||
|
PFR_Strike strike;
|
||
|
|
||
|
character = &phys->chars[glyph_index];
|
||
|
|
||
|
/* Look-up a bitmap strike corresponding to the current */
|
||
|
/* character dimensions */
|
||
|
|
||
|
{
|
||
|
FT_UInt n;
|
||
|
|
||
|
strike = phys->strikes;
|
||
|
for ( n = 0; n < phys->num_strikes; n++ )
|
||
|
{
|
||
|
if ( strike->x_ppm == (FT_UInt) size->root.metrics.x_ppem &&
|
||
|
strike->y_ppm == (FT_UInt) size->root.metrics.y_ppem )
|
||
|
{
|
||
|
goto Found_Strike;
|
||
|
}
|
||
|
strike++;
|
||
|
}
|
||
|
|
||
|
/* couldn't find it */
|
||
|
return FT_Err_Invalid_Argument;
|
||
|
}
|
||
|
|
||
|
Found_Strike:
|
||
|
|
||
|
/* Now lookup the glyph's position within the file */
|
||
|
{
|
||
|
FT_UInt char_len;
|
||
|
|
||
|
char_len = 4;
|
||
|
if ( strike->flags & 1 ) char_len += 1;
|
||
|
if ( strike->flags & 2 ) char_len += 1;
|
||
|
if ( strike->flags & 4 ) char_len += 1;
|
||
|
|
||
|
/* Access data directly in the frame to speed lookups */
|
||
|
if ( FT_STREAM_SEEK( phys->bct_offset + strike->bct_offset ) ||
|
||
|
FT_FRAME_ENTER( char_len * strike->num_bitmaps ) )
|
||
|
goto Exit;
|
||
|
|
||
|
pfr_lookup_bitmap_data( stream->cursor,
|
||
|
stream->limit,
|
||
|
strike->num_bitmaps,
|
||
|
strike->flags,
|
||
|
character->char_code,
|
||
|
&gps_offset,
|
||
|
&gps_size );
|
||
|
|
||
|
FT_FRAME_EXIT();
|
||
|
|
||
|
if (gps_size == 0)
|
||
|
{
|
||
|
/* Could not find a bitmap program string for this glyph */
|
||
|
error = FT_Err_Invalid_Argument;
|
||
|
goto Exit;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* get the bitmap metrics */
|
||
|
{
|
||
|
FT_Long xpos, ypos, advance;
|
||
|
FT_UInt xsize, ysize, format;
|
||
|
FT_Byte* p;
|
||
|
|
||
|
advance = FT_MulDiv( size->root.metrics.x_ppem << 8,
|
||
|
character->advance,
|
||
|
phys->metrics_resolution );
|
||
|
|
||
|
/* XXX: handle linearHoriAdvance correctly !! */
|
||
|
|
||
|
if ( FT_STREAM_SEEK( face->header.gps_section_offset + gps_offset ) ||
|
||
|
FT_FRAME_ENTER( gps_size ) )
|
||
|
goto Exit;
|
||
|
|
||
|
p = stream->cursor;
|
||
|
error = pfr_load_bitmap_metrics( &p, stream->limit,
|
||
|
advance,
|
||
|
&xpos, &ypos,
|
||
|
&xsize, &ysize,
|
||
|
&advance, &format );
|
||
|
if ( !error )
|
||
|
{
|
||
|
glyph->root.format = FT_GLYPH_FORMAT_BITMAP;
|
||
|
|
||
|
/* Set up glyph bitmap and metrics */
|
||
|
glyph->root.bitmap.width = (FT_Int) xsize;
|
||
|
glyph->root.bitmap.rows = (FT_Int) ysize;
|
||
|
glyph->root.bitmap.pitch = (FT_Long)(xsize+7) >> 3;
|
||
|
glyph->root.bitmap.pixel_mode = FT_PIXEL_MODE_MONO;
|
||
|
|
||
|
glyph->root.metrics.width = (FT_Long)xsize << 6;
|
||
|
glyph->root.metrics.height = (FT_Long)ysize << 6;
|
||
|
glyph->root.metrics.horiBearingX = xpos << 6;
|
||
|
glyph->root.metrics.horiBearingY = ypos << 6;
|
||
|
glyph->root.metrics.horiAdvance = ((advance >> 2) + 32) & -64;
|
||
|
glyph->root.metrics.vertBearingX = - glyph->root.metrics.width >> 1;
|
||
|
glyph->root.metrics.vertBearingY = 0;
|
||
|
glyph->root.metrics.vertAdvance = size->root.metrics.height;
|
||
|
|
||
|
glyph->root.bitmap_left = xpos;
|
||
|
glyph->root.bitmap_top = ypos + ysize;
|
||
|
|
||
|
/* Allocate and read bitmap data */
|
||
|
{
|
||
|
FT_Memory memory = face->root.memory;
|
||
|
FT_Long len = glyph->root.bitmap.pitch*ysize;
|
||
|
|
||
|
if ( !FT_ALLOC( glyph->root.bitmap.buffer, len ) )
|
||
|
{
|
||
|
error = pfr_load_bitmap_bits( p,
|
||
|
stream->limit,
|
||
|
format,
|
||
|
(face->header.color_flags & 2),
|
||
|
&glyph->root.bitmap );
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
FT_FRAME_EXIT();
|
||
|
}
|
||
|
|
||
|
Exit:
|
||
|
return error;
|
||
|
}
|