shithub: freetype+ttf2subf

ref: 072f7f89d3e6e3391cb2627cfceb3068a29c13cc
dir: /src/pfr/pfrsbit.c/

View raw version
#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;
  }