Blame libdjvu/JB2EncodeCodec.cpp

Packit df99a1
//C-  -*- C++ -*-
Packit df99a1
//C- -------------------------------------------------------------------
Packit df99a1
//C- DjVuLibre-3.5
Packit df99a1
//C- Copyright (c) 2002  Leon Bottou and Yann Le Cun.
Packit df99a1
//C- Copyright (c) 2001  AT&T
Packit df99a1
//C-
Packit df99a1
//C- This software is subject to, and may be distributed under, the
Packit df99a1
//C- GNU General Public License, either Version 2 of the license,
Packit df99a1
//C- or (at your option) any later version. The license should have
Packit df99a1
//C- accompanied the software or you may obtain a copy of the license
Packit df99a1
//C- from the Free Software Foundation at http://www.fsf.org .
Packit df99a1
//C-
Packit df99a1
//C- This program is distributed in the hope that it will be useful,
Packit df99a1
//C- but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit df99a1
//C- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit df99a1
//C- GNU General Public License for more details.
Packit df99a1
//C- 
Packit df99a1
//C- DjVuLibre-3.5 is derived from the DjVu(r) Reference Library from
Packit df99a1
//C- Lizardtech Software.  Lizardtech Software has authorized us to
Packit df99a1
//C- replace the original DjVu(r) Reference Library notice by the following
Packit df99a1
//C- text (see doc/lizard2002.djvu and doc/lizardtech2007.djvu):
Packit df99a1
//C-
Packit df99a1
//C-  ------------------------------------------------------------------
Packit df99a1
//C- | DjVu (r) Reference Library (v. 3.5)
Packit df99a1
//C- | Copyright (c) 1999-2001 LizardTech, Inc. All Rights Reserved.
Packit df99a1
//C- | The DjVu Reference Library is protected by U.S. Pat. No.
Packit df99a1
//C- | 6,058,214 and patents pending.
Packit df99a1
//C- |
Packit df99a1
//C- | This software is subject to, and may be distributed under, the
Packit df99a1
//C- | GNU General Public License, either Version 2 of the license,
Packit df99a1
//C- | or (at your option) any later version. The license should have
Packit df99a1
//C- | accompanied the software or you may obtain a copy of the license
Packit df99a1
//C- | from the Free Software Foundation at http://www.fsf.org .
Packit df99a1
//C- |
Packit df99a1
//C- | The computer code originally released by LizardTech under this
Packit df99a1
//C- | license and unmodified by other parties is deemed "the LIZARDTECH
Packit df99a1
//C- | ORIGINAL CODE."  Subject to any third party intellectual property
Packit df99a1
//C- | claims, LizardTech grants recipient a worldwide, royalty-free, 
Packit df99a1
//C- | non-exclusive license to make, use, sell, or otherwise dispose of 
Packit df99a1
//C- | the LIZARDTECH ORIGINAL CODE or of programs derived from the 
Packit df99a1
//C- | LIZARDTECH ORIGINAL CODE in compliance with the terms of the GNU 
Packit df99a1
//C- | General Public License.   This grant only confers the right to 
Packit df99a1
//C- | infringe patent claims underlying the LIZARDTECH ORIGINAL CODE to 
Packit df99a1
//C- | the extent such infringement is reasonably necessary to enable 
Packit df99a1
//C- | recipient to make, have made, practice, sell, or otherwise dispose 
Packit df99a1
//C- | of the LIZARDTECH ORIGINAL CODE (or portions thereof) and not to 
Packit df99a1
//C- | any greater extent that may be necessary to utilize further 
Packit df99a1
//C- | modifications or combinations.
Packit df99a1
//C- |
Packit df99a1
//C- | The LIZARDTECH ORIGINAL CODE is provided "AS IS" WITHOUT WARRANTY
Packit df99a1
//C- | OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
Packit df99a1
//C- | TO ANY WARRANTY OF NON-INFRINGEMENT, OR ANY IMPLIED WARRANTY OF
Packit df99a1
//C- | MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE.
Packit df99a1
//C- +------------------------------------------------------------------
Packit df99a1
Packit df99a1
#ifdef HAVE_CONFIG_H
Packit df99a1
# include "config.h"
Packit df99a1
#endif
Packit df99a1
#if NEED_GNUG_PRAGMAS
Packit df99a1
# pragma implementation
Packit df99a1
#endif
Packit df99a1
Packit df99a1
// From: Leon Bottou, 1/31/2002
Packit df99a1
// Lizardtech has split the corresponding cpp file into a decoder and an encoder.
Packit df99a1
// Only superficial changes.  The meat is mine.
Packit df99a1
Packit df99a1
#ifndef NEED_DECODER_ONLY
Packit df99a1
Packit df99a1
#include "JB2Image.h"
Packit df99a1
#include "GBitmap.h"
Packit df99a1
#include <string.h>
Packit df99a1
Packit df99a1
Packit df99a1
#ifdef HAVE_NAMESPACES
Packit df99a1
namespace DJVU {
Packit df99a1
# ifdef NOT_DEFINED // Just to fool emacs c++ mode
Packit df99a1
}
Packit df99a1
#endif
Packit df99a1
#endif
Packit df99a1
Packit df99a1
////////////////////////////////////////
Packit df99a1
//// CLASS JB2Codec::Encode:  DECLARATION
Packit df99a1
////////////////////////////////////////
Packit df99a1
Packit df99a1
// This class is accessed via the encode
Packit df99a1
// functions of class JB2Image
Packit df99a1
Packit df99a1
Packit df99a1
//**** Class JB2Codec
Packit df99a1
// This class implements the JB2 coder.
Packit df99a1
// Contains all contextual information for encoding a JB2Image.
Packit df99a1
Packit df99a1
class JB2Dict::JB2Codec::Encode : public JB2Dict::JB2Codec
Packit df99a1
{
Packit df99a1
public:
Packit df99a1
  Encode(void);
Packit df99a1
  void init(const GP<ByteStream> &gbs;;
Packit df99a1
//virtual
Packit df99a1
  void code(const GP<JB2Image> &jim);
Packit df99a1
  void code(JB2Image *jim) { const GP<JB2Image> gjim(jim);code(gjim); }
Packit df99a1
  void code(const GP<JB2Dict> &jim);
Packit df99a1
  void code(JB2Dict *jim) { const GP<JB2Dict> gjim(jim);code(gjim); }
Packit df99a1
Packit df99a1
protected:
Packit df99a1
  void CodeNum(const int num, const int lo, const int hi, NumContext &ctx;;
Packit df99a1
  void encode_libonly_shape(const GP<JB2Image> &jim, int shapeno);
Packit df99a1
// virtual
Packit df99a1
  bool CodeBit(const bool bit, BitContext &ctx;;
Packit df99a1
  void code_comment(GUTF8String &comment);
Packit df99a1
  void code_record_type(int &rectype);
Packit df99a1
  int code_match_index(int &index, JB2Dict &jim);
Packit df99a1
  void code_inherited_shape_count(JB2Dict &jim);
Packit df99a1
  void code_image_size(JB2Dict &jim);
Packit df99a1
  void code_image_size(JB2Image &jim);
Packit df99a1
  void code_absolute_location(JB2Blit *jblt,  int rows, int columns);
Packit df99a1
  void code_absolute_mark_size(GBitmap &bm, int border=0);
Packit df99a1
  void code_relative_mark_size(GBitmap &bm, int cw, int ch, int border=0);
Packit df99a1
  void code_bitmap_directly(GBitmap &bm,const int dw, int dy,
Packit df99a1
    unsigned char *up2, unsigned char *up1, unsigned char *up0 );
Packit df99a1
  int get_diff(const int x_diff,NumContext &rel_loc);
Packit df99a1
  void code_bitmap_by_cross_coding (GBitmap &bm, GBitmap &cbm,
Packit df99a1
    const int xd2c, const int dw, int dy, int cy,
Packit df99a1
    unsigned char *up1, unsigned char *up0, unsigned char *xup1, 
Packit df99a1
    unsigned char *xup0, unsigned char *xdn1 );
Packit df99a1
Packit df99a1
private:
Packit df99a1
  GP<ZPCodec> gzp;
Packit df99a1
};
Packit df99a1
Packit df99a1
Packit df99a1
////////////////////////////////////////
Packit df99a1
//// CLASS JB2DICT: IMPLEMENTATION
Packit df99a1
////////////////////////////////////////
Packit df99a1
Packit df99a1
void 
Packit df99a1
JB2Dict::encode(const GP<ByteStream> &gbs) const
Packit df99a1
{
Packit df99a1
  JB2Codec::Encode codec;
Packit df99a1
  codec.init(gbs);
Packit df99a1
  codec.code(const_cast<JB2Dict *>(this));
Packit df99a1
}
Packit df99a1
Packit df99a1
////////////////////////////////////////
Packit df99a1
//// CLASS JB2IMAGE: IMPLEMENTATION
Packit df99a1
////////////////////////////////////////
Packit df99a1
Packit df99a1
void 
Packit df99a1
JB2Image::encode(const GP<ByteStream> &gbs) const
Packit df99a1
{
Packit df99a1
  JB2Codec::Encode codec;
Packit df99a1
  codec.init(gbs);
Packit df99a1
  codec.code(const_cast<JB2Image *>(this));
Packit df99a1
}
Packit df99a1
Packit df99a1
////////////////////////////////////////
Packit df99a1
//// CLASS JB2CODEC : IMPLEMENTATION
Packit df99a1
////////////////////////////////////////
Packit df99a1
Packit df99a1
#define START_OF_DATA                   (0)
Packit df99a1
#define NEW_MARK                        (1)
Packit df99a1
#define NEW_MARK_LIBRARY_ONLY           (2)
Packit df99a1
#define NEW_MARK_IMAGE_ONLY             (3)
Packit df99a1
#define MATCHED_REFINE                  (4)
Packit df99a1
#define MATCHED_REFINE_LIBRARY_ONLY     (5)
Packit df99a1
#define MATCHED_REFINE_IMAGE_ONLY       (6)
Packit df99a1
#define MATCHED_COPY                    (7)
Packit df99a1
#define NON_MARK_DATA                   (8)
Packit df99a1
#define REQUIRED_DICT_OR_RESET          (9)
Packit df99a1
#define PRESERVED_COMMENT               (10)
Packit df99a1
#define END_OF_DATA                     (11)
Packit df99a1
Packit df99a1
// STATIC DATA MEMBERS
Packit df99a1
Packit df99a1
static const int BIGPOSITIVE = 262142;
Packit df99a1
static const int BIGNEGATIVE = -262143;
Packit df99a1
static const int CELLCHUNK = 20000;
Packit df99a1
Packit df99a1
// CONSTRUCTOR
Packit df99a1
Packit df99a1
JB2Dict::JB2Codec::Encode::Encode(void)
Packit df99a1
: JB2Dict::JB2Codec(1) {}
Packit df99a1
Packit df99a1
void
Packit df99a1
JB2Dict::JB2Codec::Encode::init(const GP<ByteStream> &gbs)
Packit df99a1
{
Packit df99a1
  gzp=ZPCodec::create(gbs,true,true);
Packit df99a1
}
Packit df99a1
Packit df99a1
inline bool
Packit df99a1
JB2Dict::JB2Codec::Encode::CodeBit(const bool bit, BitContext &ctx)
Packit df99a1
{
Packit df99a1
    gzp->encoder(bit?1:0, ctx);
Packit df99a1
    return bit;
Packit df99a1
}
Packit df99a1
Packit df99a1
void
Packit df99a1
JB2Dict::JB2Codec::Encode::CodeNum(int num, int low, int high, NumContext &ctx)
Packit df99a1
{
Packit df99a1
  if (num < low || num > high)
Packit df99a1
    G_THROW( ERR_MSG("JB2Image.bad_number") );
Packit df99a1
  JB2Codec::CodeNum(low,high,&ctx,num);
Packit df99a1
}
Packit df99a1
Packit df99a1
// CODE COMMENTS
Packit df99a1
Packit df99a1
void 
Packit df99a1
JB2Dict::JB2Codec::Encode::code_comment(GUTF8String &comment)
Packit df99a1
{
Packit df99a1
  // Encode size
Packit df99a1
      int size=comment.length();
Packit df99a1
      CodeNum(size, 0, BIGPOSITIVE, dist_comment_length);
Packit df99a1
      for (int i=0; i
Packit df99a1
        {
Packit df99a1
          CodeNum(comment[i], 0, 255, dist_comment_byte);
Packit df99a1
        }
Packit df99a1
}
Packit df99a1
Packit df99a1
// CODE SIMPLE VALUES
Packit df99a1
Packit df99a1
inline void 
Packit df99a1
JB2Dict::JB2Codec::Encode::code_record_type(int &rectype)
Packit df99a1
{
Packit df99a1
  CodeNum(rectype, START_OF_DATA, END_OF_DATA, dist_record_type);
Packit df99a1
}
Packit df99a1
Packit df99a1
int 
Packit df99a1
JB2Dict::JB2Codec::Encode::code_match_index(int &index, JB2Dict &jim)
Packit df99a1
{
Packit df99a1
    int match=shape2lib[index];
Packit df99a1
    CodeNum(match, 0, lib2shape.hbound(), dist_match_index);
Packit df99a1
    return match;
Packit df99a1
}
Packit df99a1
Packit df99a1
// CODE PAIRS
Packit df99a1
Packit df99a1
void
Packit df99a1
JB2Dict::JB2Codec::Encode::code_inherited_shape_count(JB2Dict &jim)
Packit df99a1
{
Packit df99a1
  CodeNum(jim.get_inherited_shape_count(),
Packit df99a1
    0, BIGPOSITIVE, inherited_shape_count_dist);
Packit df99a1
}
Packit df99a1
Packit df99a1
void 
Packit df99a1
JB2Dict::JB2Codec::Encode::code_image_size(JB2Dict &jim)
Packit df99a1
{
Packit df99a1
  CodeNum(0, 0, BIGPOSITIVE, image_size_dist);
Packit df99a1
  CodeNum(0, 0, BIGPOSITIVE, image_size_dist);
Packit df99a1
  JB2Codec::code_image_size(jim);
Packit df99a1
}
Packit df99a1
Packit df99a1
void 
Packit df99a1
JB2Dict::JB2Codec::Encode::code_image_size(JB2Image &jim)
Packit df99a1
{
Packit df99a1
  image_columns = jim.get_width();
Packit df99a1
  CodeNum(image_columns, 0, BIGPOSITIVE, image_size_dist);
Packit df99a1
  image_rows = jim.get_height();
Packit df99a1
  CodeNum(image_rows, 0, BIGPOSITIVE, image_size_dist);
Packit df99a1
  JB2Codec::code_image_size(jim);
Packit df99a1
}
Packit df99a1
Packit df99a1
inline int
Packit df99a1
JB2Dict::JB2Codec::Encode::get_diff(int x_diff,NumContext &rel_loc)
Packit df99a1
{
Packit df99a1
   CodeNum(x_diff, BIGNEGATIVE, BIGPOSITIVE, rel_loc);
Packit df99a1
   return x_diff;
Packit df99a1
}
Packit df99a1
Packit df99a1
void 
Packit df99a1
JB2Dict::JB2Codec::Encode::code_absolute_location(JB2Blit *jblt, int rows, int columns)
Packit df99a1
{
Packit df99a1
  // Check start record
Packit df99a1
  if (!gotstartrecordp)
Packit df99a1
    G_THROW( ERR_MSG("JB2Image.no_start") );
Packit df99a1
  // Code TOP and LEFT
Packit df99a1
  CodeNum(jblt->left+1, 1, image_columns, abs_loc_x);
Packit df99a1
  CodeNum(jblt->bottom+rows-1+1, 1, image_rows, abs_loc_y);
Packit df99a1
}
Packit df99a1
Packit df99a1
void 
Packit df99a1
JB2Dict::JB2Codec::Encode::code_absolute_mark_size(GBitmap &bm, int border)
Packit df99a1
{
Packit df99a1
  CodeNum(bm.columns(), 0, BIGPOSITIVE, abs_size_x);
Packit df99a1
  CodeNum(bm.rows(), 0, BIGPOSITIVE, abs_size_y);
Packit df99a1
}
Packit df99a1
Packit df99a1
void 
Packit df99a1
JB2Dict::JB2Codec::Encode::code_relative_mark_size(GBitmap &bm, int cw, int ch, int border)
Packit df99a1
{
Packit df99a1
  CodeNum(bm.columns()-cw, BIGNEGATIVE, BIGPOSITIVE, rel_size_x);
Packit df99a1
  CodeNum(bm.rows()-ch, BIGNEGATIVE, BIGPOSITIVE, rel_size_y);
Packit df99a1
}
Packit df99a1
Packit df99a1
// CODE BITMAP DIRECTLY
Packit df99a1
Packit df99a1
void 
Packit df99a1
JB2Dict::JB2Codec::Encode::code_bitmap_directly(
Packit df99a1
  GBitmap &bm,const int dw, int dy,
Packit df99a1
  unsigned char *up2, unsigned char *up1, unsigned char *up0 )
Packit df99a1
{
Packit df99a1
      ZPCodec &zp=*gzp;
Packit df99a1
      // iterate on rows (encoding)
Packit df99a1
      while (dy >= 0)
Packit df99a1
        {
Packit df99a1
          int context=get_direct_context(up2, up1, up0, 0);
Packit df99a1
          for (int dx=0;dx < dw;)
Packit df99a1
            {
Packit df99a1
              int n = up0[dx++];
Packit df99a1
              zp.encoder(n, bitdist[context]);
Packit df99a1
              context=shift_direct_context(context, n, up2, up1, up0, dx);
Packit df99a1
            }
Packit df99a1
          // next row
Packit df99a1
          dy -= 1;
Packit df99a1
          up2 = up1;
Packit df99a1
          up1 = up0;
Packit df99a1
          up0 = bm[dy];
Packit df99a1
        }
Packit df99a1
}
Packit df99a1
Packit df99a1
// CODE BITMAP BY CROSS CODING
Packit df99a1
Packit df99a1
void 
Packit df99a1
JB2Dict::JB2Codec::Encode::code_bitmap_by_cross_coding (GBitmap &bm, GBitmap &cbm,
Packit df99a1
  const int xd2c, const int dw, int dy, int cy,
Packit df99a1
  unsigned char *up1, unsigned char *up0, unsigned char *xup1, 
Packit df99a1
  unsigned char *xup0, unsigned char *xdn1 )
Packit df99a1
{
Packit df99a1
      ZPCodec &zp=*gzp;
Packit df99a1
      // iterate on rows (encoding)
Packit df99a1
      while (dy >= 0)
Packit df99a1
        {
Packit df99a1
          int context=get_cross_context(up1, up0, xup1, xup0, xdn1, 0);
Packit df99a1
          for(int dx=0;dx < dw;)
Packit df99a1
            {
Packit df99a1
              const int n = up0[dx++];
Packit df99a1
              zp.encoder(n, cbitdist[context]);
Packit df99a1
              context=shift_cross_context(context, n,  
Packit df99a1
                                  up1, up0, xup1, xup0, xdn1, dx);
Packit df99a1
            }
Packit df99a1
          // next row
Packit df99a1
          up1 = up0;
Packit df99a1
          up0 = bm[--dy];
Packit df99a1
          xup1 = xup0;
Packit df99a1
          xup0 = xdn1;
Packit df99a1
          xdn1 = cbm[(--cy)-1] + xd2c;
Packit df99a1
        }
Packit df99a1
}
Packit df99a1
Packit df99a1
// CODE JB2DICT
Packit df99a1
Packit df99a1
void 
Packit df99a1
JB2Dict::JB2Codec::Encode::code(const GP<JB2Dict> &gjim)
Packit df99a1
{
Packit df99a1
  if(!gjim)
Packit df99a1
  {
Packit df99a1
    G_THROW( ERR_MSG("JB2Image.bad_number") );
Packit df99a1
  }
Packit df99a1
  JB2Dict &jim=*gjim;
Packit df99a1
      // -------------------------
Packit df99a1
      // THIS IS THE ENCODING PART
Packit df99a1
      // -------------------------
Packit df99a1
      int firstshape = jim.get_inherited_shape_count();
Packit df99a1
      int nshape = jim.get_shape_count();
Packit df99a1
      init_library(jim);
Packit df99a1
      // Code headers.
Packit df99a1
      int rectype = REQUIRED_DICT_OR_RESET;
Packit df99a1
      if (jim.get_inherited_shape_count() > 0)
Packit df99a1
        code_record(rectype, gjim, 0);
Packit df99a1
      rectype = START_OF_DATA;
Packit df99a1
      code_record(rectype, gjim, 0);
Packit df99a1
      // Code Comment.
Packit df99a1
      rectype = PRESERVED_COMMENT;
Packit df99a1
      if (!! jim.comment)
Packit df99a1
        code_record(rectype, gjim, 0);
Packit df99a1
      // Encode every shape
Packit df99a1
      int shapeno;
Packit df99a1
      DJVU_PROGRESS_TASK(jb2code,"jb2 encode", nshape-firstshape);
Packit df99a1
      for (shapeno=firstshape; shapeno
Packit df99a1
        {
Packit df99a1
          DJVU_PROGRESS_RUN(jb2code, (shapeno-firstshape)|0xff);
Packit df99a1
          // Code shape
Packit df99a1
          JB2Shape &jshp = jim.get_shape(shapeno);
Packit df99a1
          rectype=(jshp.parent >= 0)
Packit df99a1
            ?MATCHED_REFINE_LIBRARY_ONLY:NEW_MARK_LIBRARY_ONLY;
Packit df99a1
          code_record(rectype, gjim, &jshp);
Packit df99a1
          add_library(shapeno, jshp);
Packit df99a1
	  // Check numcoder status
Packit df99a1
	  if (cur_ncell > CELLCHUNK) 
Packit df99a1
	    {
Packit df99a1
	      rectype = REQUIRED_DICT_OR_RESET;
Packit df99a1
	      code_record(rectype, 0, 0);	      
Packit df99a1
	    }
Packit df99a1
        }
Packit df99a1
      // Code end of data record
Packit df99a1
      rectype = END_OF_DATA;
Packit df99a1
      code_record(rectype, gjim, 0); 
Packit df99a1
      gzp=0;
Packit df99a1
}
Packit df99a1
Packit df99a1
// CODE JB2IMAGE
Packit df99a1
Packit df99a1
void 
Packit df99a1
JB2Dict::JB2Codec::Encode::code(const GP<JB2Image> &gjim)
Packit df99a1
{
Packit df99a1
  if(!gjim)
Packit df99a1
  {
Packit df99a1
    G_THROW( ERR_MSG("JB2Image.bad_number") );
Packit df99a1
  }
Packit df99a1
  JB2Image &jim=*gjim;
Packit df99a1
      // -------------------------
Packit df99a1
      // THIS IS THE ENCODING PART
Packit df99a1
      // -------------------------
Packit df99a1
      int i;
Packit df99a1
      init_library(jim);
Packit df99a1
      int firstshape = jim.get_inherited_shape_count();
Packit df99a1
      int nshape = jim.get_shape_count();
Packit df99a1
      int nblit = jim.get_blit_count();
Packit df99a1
      // Initialize shape2lib 
Packit df99a1
      shape2lib.resize(0,nshape-1);
Packit df99a1
      for (i=firstshape; i
Packit df99a1
        shape2lib[i] = -1;
Packit df99a1
      // Determine shapes that go into library (shapeno>=firstshape)
Packit df99a1
      //  shape2lib is -2 if used by one blit
Packit df99a1
      //  shape2lib is -3 if used by more than one blit
Packit df99a1
      //  shape2lib is -4 if used as a parent
Packit df99a1
      for (i=0; i
Packit df99a1
        {
Packit df99a1
          JB2Blit *jblt = jim.get_blit(i);
Packit df99a1
          int shapeno = jblt->shapeno;
Packit df99a1
          if (shapeno < firstshape)
Packit df99a1
            continue;
Packit df99a1
          if (shape2lib[shapeno] >= -2) 
Packit df99a1
            shape2lib[shapeno] -= 1;
Packit df99a1
          shapeno = jim.get_shape(shapeno).parent;
Packit df99a1
          while (shapeno>=firstshape && shape2lib[shapeno]>=-3)
Packit df99a1
            {
Packit df99a1
              shape2lib[shapeno] = -4;
Packit df99a1
              shapeno = jim.get_shape(shapeno).parent;
Packit df99a1
            }
Packit df99a1
        }
Packit df99a1
      // Code headers.
Packit df99a1
      int rectype = REQUIRED_DICT_OR_RESET;
Packit df99a1
      if (jim.get_inherited_shape_count() > 0)
Packit df99a1
        code_record(rectype, gjim, 0, 0);
Packit df99a1
      rectype = START_OF_DATA;
Packit df99a1
      code_record(rectype, gjim, 0, 0);
Packit df99a1
      // Code Comment.
Packit df99a1
      rectype = PRESERVED_COMMENT;
Packit df99a1
      if (!! jim.comment)
Packit df99a1
        code_record(rectype, gjim, 0, 0);
Packit df99a1
      // Encode every blit
Packit df99a1
      int blitno;
Packit df99a1
      DJVU_PROGRESS_TASK(jb2code,"jb2 encode", nblit);
Packit df99a1
      for (blitno=0; blitno
Packit df99a1
        {
Packit df99a1
          DJVU_PROGRESS_RUN(jb2code, blitno|0xff);
Packit df99a1
          JB2Blit *jblt = jim.get_blit(blitno);
Packit df99a1
          int shapeno = jblt->shapeno;
Packit df99a1
          JB2Shape &jshp = jim.get_shape(shapeno);
Packit df99a1
          // Tests if shape exists in library
Packit df99a1
          if (shape2lib[shapeno] >= 0)
Packit df99a1
            {
Packit df99a1
              int rectype = MATCHED_COPY;
Packit df99a1
              code_record(rectype, gjim, 0, jblt);
Packit df99a1
            }
Packit df99a1
          // Avoid coding null shapes/blits
Packit df99a1
          else if (jshp.bits) 
Packit df99a1
            {
Packit df99a1
              // Make sure all parents have been coded
Packit df99a1
              if (jshp.parent>=0 && shape2lib[jshp.parent]<0)
Packit df99a1
                encode_libonly_shape(gjim, jshp.parent);
Packit df99a1
              // Allocate library entry when needed
Packit df99a1
#define LIBRARY_CONTAINS_ALL
Packit df99a1
              int libraryp = 0;
Packit df99a1
#ifdef LIBRARY_CONTAINS_MARKS // baseline
Packit df99a1
              if (jshp.parent >= -1)
Packit df99a1
                libraryp = 1;
Packit df99a1
#endif
Packit df99a1
#ifdef LIBRARY_CONTAINS_SHARED // worse             
Packit df99a1
              if (shape2lib[shapeno] <= -3)
Packit df99a1
                libraryp = 1;
Packit df99a1
#endif
Packit df99a1
#ifdef LIBRARY_CONTAINS_ALL // better
Packit df99a1
              libraryp = 1;
Packit df99a1
#endif
Packit df99a1
              // Test all blit cases
Packit df99a1
              if (jshp.parent<-1 && !libraryp)
Packit df99a1
                {
Packit df99a1
                  int rectype = NON_MARK_DATA;
Packit df99a1
                  code_record(rectype, gjim, &jshp, jblt);
Packit df99a1
                }
Packit df99a1
              else if (jshp.parent < 0)
Packit df99a1
                {
Packit df99a1
                  int rectype = (libraryp ? NEW_MARK : NEW_MARK_IMAGE_ONLY);
Packit df99a1
                  code_record(rectype, gjim, &jshp, jblt);
Packit df99a1
                }
Packit df99a1
              else 
Packit df99a1
                {
Packit df99a1
                  int rectype = (libraryp ? MATCHED_REFINE : MATCHED_REFINE_IMAGE_ONLY);
Packit df99a1
                  code_record(rectype, gjim, &jshp, jblt);
Packit df99a1
                }
Packit df99a1
              // Add shape to library
Packit df99a1
              if (libraryp) 
Packit df99a1
                add_library(shapeno, jshp);
Packit df99a1
            }
Packit df99a1
	  // Check numcoder status
Packit df99a1
	  if (cur_ncell > CELLCHUNK) 
Packit df99a1
	    {
Packit df99a1
	      rectype = REQUIRED_DICT_OR_RESET;
Packit df99a1
	      code_record(rectype, 0, 0);
Packit df99a1
	    }
Packit df99a1
        }
Packit df99a1
      // Code end of data record
Packit df99a1
      rectype = END_OF_DATA;
Packit df99a1
      code_record(rectype, gjim, 0, 0); 
Packit df99a1
      gzp=0;
Packit df99a1
}
Packit df99a1
Packit df99a1
////////////////////////////////////////
Packit df99a1
//// HELPERS
Packit df99a1
////////////////////////////////////////
Packit df99a1
Packit df99a1
void 
Packit df99a1
JB2Dict::JB2Codec::Encode::encode_libonly_shape(
Packit df99a1
  const GP<JB2Image> &gjim, int shapeno )
Packit df99a1
{
Packit df99a1
  if(!gjim)
Packit df99a1
  {
Packit df99a1
    G_THROW( ERR_MSG("JB2Image.bad_number") );
Packit df99a1
  }
Packit df99a1
  JB2Image &jim=*gjim;
Packit df99a1
  // Recursively encode parent shape
Packit df99a1
  JB2Shape &jshp = jim.get_shape(shapeno);
Packit df99a1
  if (jshp.parent>=0 && shape2lib[jshp.parent]<0)
Packit df99a1
    encode_libonly_shape(gjim, jshp.parent);
Packit df99a1
  // Test that library shape must be encoded
Packit df99a1
  if (shape2lib[shapeno] < 0)
Packit df99a1
    {
Packit df99a1
      // Code library entry
Packit df99a1
      int rectype=(jshp.parent >= 0)
Packit df99a1
            ?NEW_MARK_LIBRARY_ONLY:MATCHED_REFINE_LIBRARY_ONLY;
Packit df99a1
      code_record(rectype, gjim, &jshp, 0);      
Packit df99a1
      // Add shape to library
Packit df99a1
      add_library(shapeno, jshp);
Packit df99a1
      // Check numcoder status
Packit df99a1
      if (cur_ncell > CELLCHUNK) 
Packit df99a1
	{
Packit df99a1
	  rectype = REQUIRED_DICT_OR_RESET;
Packit df99a1
	  code_record(rectype, 0, 0);
Packit df99a1
	}
Packit df99a1
    }
Packit df99a1
}
Packit df99a1
Packit df99a1
Packit df99a1
#ifdef HAVE_NAMESPACES
Packit df99a1
}
Packit df99a1
# ifndef NOT_USING_DJVU_NAMESPACE
Packit df99a1
using namespace DJVU;
Packit df99a1
# endif
Packit df99a1
#endif
Packit df99a1
Packit df99a1
#endif /* NEED_DECODER_ONLY */
Packit df99a1