Blame src/lzo1b_c.ch

Packit Service 5195f2
/* lzo1b_c.ch -- implementation of the LZO1B compression algorithm
Packit Service 5195f2
Packit Service 5195f2
   This file is part of the LZO real-time data compression library.
Packit Service 5195f2
Packit Service 5195f2
   Copyright (C) 1996-2014 Markus Franz Xaver Johannes Oberhumer
Packit Service 5195f2
   All Rights Reserved.
Packit Service 5195f2
Packit Service 5195f2
   The LZO library is free software; you can redistribute it and/or
Packit Service 5195f2
   modify it under the terms of the GNU General Public License as
Packit Service 5195f2
   published by the Free Software Foundation; either version 2 of
Packit Service 5195f2
   the License, or (at your option) any later version.
Packit Service 5195f2
Packit Service 5195f2
   The LZO library is distributed in the hope that it will be useful,
Packit Service 5195f2
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 5195f2
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit Service 5195f2
   GNU General Public License for more details.
Packit Service 5195f2
Packit Service 5195f2
   You should have received a copy of the GNU General Public License
Packit Service 5195f2
   along with the LZO library; see the file COPYING.
Packit Service 5195f2
   If not, write to the Free Software Foundation, Inc.,
Packit Service 5195f2
   51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
Packit Service 5195f2
Packit Service 5195f2
   Markus F.X.J. Oberhumer
Packit Service 5195f2
   <markus@oberhumer.com>
Packit Service 5195f2
   http://www.oberhumer.com/opensource/lzo/
Packit Service 5195f2
 */
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
/***********************************************************************
Packit Service 5195f2
//
Packit Service 5195f2
************************************************************************/
Packit Service 5195f2
Packit Service 5195f2
#if !defined(LZO_HAVE_R1) && !defined(LZO_NO_R1)
Packit Service 5195f2
#  define LZO_HAVE_R1 1
Packit Service 5195f2
#endif
Packit Service 5195f2
Packit Service 5195f2
#if !defined(LZO_HAVE_M3) && !defined(LZO_NO_M3)
Packit Service 5195f2
#  if (M3O_BITS < 8)
Packit Service 5195f2
#    define LZO_HAVE_M3 1
Packit Service 5195f2
#  endif
Packit Service 5195f2
#endif
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
#define MI      /*empty*/
Packit Service 5195f2
#define SI      MI
Packit Service 5195f2
#if (DD_BITS > 0)
Packit Service 5195f2
#define DI      ++ii; DVAL_NEXT(dv,ii); UPDATE_D(dict,drun,dv,ii,in); MI
Packit Service 5195f2
#define XI      assert(ii < ip); ii = ip; DVAL_FIRST(dv,(ip));
Packit Service 5195f2
#else
Packit Service 5195f2
#define DI      ++ii; DINDEX1(dindex,ii); UPDATE_I(dict,0,dindex,ii,in); MI
Packit Service 5195f2
#define XI      assert(ii < ip); ii = ip;
Packit Service 5195f2
#endif
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
/***********************************************************************
Packit Service 5195f2
// compress a block of data.
Packit Service 5195f2
//
Packit Service 5195f2
// I really apologize for this spaghetti code.
Packit Service 5195f2
************************************************************************/
Packit Service 5195f2
Packit Service 5195f2
#ifdef __cplusplus
Packit Service 5195f2
extern "C" {
Packit Service 5195f2
#endif
Packit Service 5195f2
LZO_PRIVATE(int)
Packit Service 5195f2
do_compress    ( const lzo_bytep in , lzo_uint  in_len,
Packit Service 5195f2
                       lzo_bytep out, lzo_uintp out_len,
Packit Service 5195f2
                       lzo_voidp wrkmem )
Packit Service 5195f2
{
Packit Service 5195f2
    const lzo_bytep ip;
Packit Service 5195f2
#if (DD_BITS > 0)
Packit Service 5195f2
#if defined(__LZO_HASH_INCREMENTAL)
Packit Service 5195f2
    lzo_xint dv;
Packit Service 5195f2
#endif
Packit Service 5195f2
    unsigned drun = 0;
Packit Service 5195f2
#endif
Packit Service 5195f2
    lzo_bytep op;
Packit Service 5195f2
    const lzo_bytep const in_end = in + in_len;
Packit Service 5195f2
    const lzo_bytep const ip_end = in + in_len - MIN_LOOKAHEAD;
Packit Service 5195f2
    const lzo_bytep ii;
Packit Service 5195f2
#if defined(LZO_HAVE_R1)
Packit Service 5195f2
    const lzo_bytep r1 = ip_end;    /* pointer for R1 match (none yet) */
Packit Service 5195f2
#endif
Packit Service 5195f2
#if defined(LZO_HAVE_M3)
Packit Service 5195f2
    lzo_bytep m3 = out + 1;         /* pointer after last m3/m4 match */
Packit Service 5195f2
#endif
Packit Service 5195f2
Packit Service 5195f2
    lzo_dict_p const dict = (lzo_dict_p) wrkmem;
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
#if (LZO_COLLECT_STATS)
Packit Service 5195f2
    lzo_stats->r_bits   = R_BITS;
Packit Service 5195f2
    lzo_stats->m3o_bits = M3O_BITS;
Packit Service 5195f2
    lzo_stats->dd_bits  = DD_BITS;
Packit Service 5195f2
    lzo_stats->clevel   = CLEVEL;
Packit Service 5195f2
    lzo_stats->d_bits   = D_BITS;
Packit Service 5195f2
    lzo_stats->min_lookahead  = MIN_LOOKAHEAD;
Packit Service 5195f2
    lzo_stats->max_lookbehind = MAX_LOOKBEHIND;
Packit Service 5195f2
    lzo_stats->compress_id    = LZO_PP_MACRO_EXPAND(COMPRESS_ID);
Packit Service 5195f2
#endif
Packit Service 5195f2
Packit Service 5195f2
    /* init dictionary */
Packit Service 5195f2
#if (LZO_DETERMINISTIC)
Packit Service 5195f2
    BZERO8_PTR(wrkmem,sizeof(lzo_dict_t),D_SIZE);
Packit Service 5195f2
#endif
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
    op = out;
Packit Service 5195f2
    ip = in;
Packit Service 5195f2
    ii = ip;            /* point to start of current literal run */
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
#if (DD_BITS > 0)
Packit Service 5195f2
    DVAL_FIRST(dv,ip);
Packit Service 5195f2
    UPDATE_D(dict,drun,dv,ip,in);
Packit Service 5195f2
    ip++;
Packit Service 5195f2
    DVAL_NEXT(dv,ip);
Packit Service 5195f2
#else
Packit Service 5195f2
    ip++;
Packit Service 5195f2
#endif
Packit Service 5195f2
Packit Service 5195f2
    assert(ip < ip_end);
Packit Service 5195f2
    for (;;)
Packit Service 5195f2
    {
Packit Service 5195f2
        const lzo_bytep m_pos;
Packit Service 5195f2
#if !defined(NDEBUG)
Packit Service 5195f2
        const lzo_bytep m_pos_sav = NULL;
Packit Service 5195f2
#endif
Packit Service 5195f2
        LZO_DEFINE_UNINITIALIZED_VAR(lzo_uint, m_off, 0);
Packit Service 5195f2
#if (DD_BITS == 0)
Packit Service 5195f2
        lzo_uint dindex;
Packit Service 5195f2
#endif
Packit Service 5195f2
        lzo_uint m_len;
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
/***********************************************************************
Packit Service 5195f2
// search for a match
Packit Service 5195f2
************************************************************************/
Packit Service 5195f2
Packit Service 5195f2
#if !defined(LZO_SEARCH_MATCH_INCLUDE_FILE)
Packit Service 5195f2
#  define LZO_SEARCH_MATCH_INCLUDE_FILE     "lzo1b_sm.ch"
Packit Service 5195f2
#endif
Packit Service 5195f2
Packit Service 5195f2
#include LZO_SEARCH_MATCH_INCLUDE_FILE
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
#if !defined(LZO_TEST_MATCH_INCLUDE_FILE)
Packit Service 5195f2
#  define LZO_TEST_MATCH_INCLUDE_FILE       "lzo1b_tm.ch"
Packit Service 5195f2
#endif
Packit Service 5195f2
Packit Service 5195f2
#include LZO_TEST_MATCH_INCLUDE_FILE
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
/***********************************************************************
Packit Service 5195f2
// found a literal
Packit Service 5195f2
************************************************************************/
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
    /* a literal */
Packit Service 5195f2
literal:
Packit Service 5195f2
#if (DD_BITS == 0)
Packit Service 5195f2
        UPDATE_I(dict,0,dindex,ip,in);
Packit Service 5195f2
#endif
Packit Service 5195f2
        if (++ip >= ip_end)
Packit Service 5195f2
            break;
Packit Service 5195f2
#if (DD_BITS > 0)
Packit Service 5195f2
        DVAL_NEXT(dv,ip);
Packit Service 5195f2
#endif
Packit Service 5195f2
        continue;
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
/***********************************************************************
Packit Service 5195f2
// found a match
Packit Service 5195f2
************************************************************************/
Packit Service 5195f2
Packit Service 5195f2
match:
Packit Service 5195f2
#if (DD_BITS == 0)
Packit Service 5195f2
        UPDATE_I(dict,0,dindex,ip,in);
Packit Service 5195f2
#endif
Packit Service 5195f2
        /* we have found a match of at least M2_MIN_LEN */
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
#if !defined(LZO_CODE_RUN_INCLUDE_FILE)
Packit Service 5195f2
#  define LZO_CODE_RUN_INCLUDE_FILE     "lzo1b_cr.ch"
Packit Service 5195f2
#endif
Packit Service 5195f2
Packit Service 5195f2
#include LZO_CODE_RUN_INCLUDE_FILE
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
        /* ii now points to the start of the current match */
Packit Service 5195f2
        assert(ii == ip);
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
/***********************************************************************
Packit Service 5195f2
// code the match
Packit Service 5195f2
************************************************************************/
Packit Service 5195f2
Packit Service 5195f2
#if !defined(LZO_CODE_MATCH_INCLUDE_FILE)
Packit Service 5195f2
#  define LZO_CODE_MATCH_INCLUDE_FILE   "lzo1b_cm.ch"
Packit Service 5195f2
#endif
Packit Service 5195f2
Packit Service 5195f2
#include LZO_CODE_MATCH_INCLUDE_FILE
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
        /* ii now points to the start of the next literal run */
Packit Service 5195f2
        assert(ii == ip);
Packit Service 5195f2
Packit Service 5195f2
    }
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
/***********************************************************************
Packit Service 5195f2
// end of block
Packit Service 5195f2
************************************************************************/
Packit Service 5195f2
Packit Service 5195f2
    assert(ip <= in_end);
Packit Service 5195f2
Packit Service 5195f2
#if (LZO_COLLECT_STATS)
Packit Service 5195f2
    {
Packit Service 5195f2
        lzo_uint i;
Packit Service 5195f2
        const lzo_bytep p;
Packit Service 5195f2
Packit Service 5195f2
        for (i = 0; i < D_SIZE; i++)
Packit Service 5195f2
        {
Packit Service 5195f2
            p = dict[i];
Packit Service 5195f2
            if (BOUNDS_CHECKING_OFF_IN_EXPR(p == NULL || p < in || p > in_end))
Packit Service 5195f2
                lzo_stats->unused_dict_entries++;
Packit Service 5195f2
        }
Packit Service 5195f2
        lzo_stats->unused_dict_entries_percent =
Packit Service 5195f2
            100.0 * lzo_stats->unused_dict_entries / D_SIZE;
Packit Service 5195f2
    }
Packit Service 5195f2
#endif
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
#if defined(LZO_RETURN_IF_NOT_COMPRESSIBLE)
Packit Service 5195f2
    /* return if op == out to indicate that we
Packit Service 5195f2
     * couldn't compress and didn't copy anything.
Packit Service 5195f2
     */
Packit Service 5195f2
    if (op == out)
Packit Service 5195f2
    {
Packit Service 5195f2
        *out_len = 0;
Packit Service 5195f2
        return LZO_E_NOT_COMPRESSIBLE;
Packit Service 5195f2
    }
Packit Service 5195f2
#endif
Packit Service 5195f2
Packit Service 5195f2
    /* store the final literal run */
Packit Service 5195f2
    if (pd(in_end,ii) > 0)
Packit Service 5195f2
    {
Packit Service 5195f2
        lzo_uint t = pd(in_end,ii);
Packit Service 5195f2
        op = STORE_RUN(op,ii,t);
Packit Service 5195f2
    }
Packit Service 5195f2
Packit Service 5195f2
    *out_len = pd(op, out);
Packit Service 5195f2
    return LZO_E_OK;                /* compression went ok */
Packit Service 5195f2
}
Packit Service 5195f2
#ifdef __cplusplus
Packit Service 5195f2
} /* extern "C" */
Packit Service 5195f2
#endif
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
/*
Packit Service 5195f2
vi:ts=4:et
Packit Service 5195f2
*/