Blame src/lzo2a_9x.c

Packit Service 5195f2
/* lzo2a_9x.c -- implementation of the LZO2A-999 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
#include "config2a.h"
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
/***********************************************************************
Packit Service 5195f2
//
Packit Service 5195f2
************************************************************************/
Packit Service 5195f2
Packit Service 5195f2
#define SWD_THRESHOLD       1           /* lower limit for match length */
Packit Service 5195f2
#define SWD_F            2048           /* upper limit for match length */
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
#define LZO2A 1
Packit Service 5195f2
#define LZO_COMPRESS_T  lzo2a_999_t
Packit Service 5195f2
#define lzo_swd_t       lzo2a_999_swd_t
Packit Service 5195f2
#include "lzo_mchw.ch"
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
#if (LZO_CC_BORLANDC && LZO_MM_FLAT)
Packit Service 5195f2
#  if ((__BORLANDC__) >= 0x0450 && (__BORLANDC__) < 0x0460)
Packit Service 5195f2
     /* avoid internal compiler error */
Packit Service 5195f2
#    pragma option -Od
Packit Service 5195f2
#  endif
Packit Service 5195f2
#endif
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
/***********************************************************************
Packit Service 5195f2
//
Packit Service 5195f2
************************************************************************/
Packit Service 5195f2
Packit Service 5195f2
#define putbyte(x)      *op++ = LZO_BYTE(x)
Packit Service 5195f2
Packit Service 5195f2
#define putbits(j,x) \
Packit Service 5195f2
    if (k == 0) bitp = op++; \
Packit Service 5195f2
    SETBITS(j,x); \
Packit Service 5195f2
    if (k >= 8) { *bitp = LZO_BYTE(MASKBITS(8)); DUMPBITS(8); \
Packit Service 5195f2
                    if (k > 0) bitp = op++; }
Packit Service 5195f2
Packit Service 5195f2
#define putbit(x)       putbits(1,x)
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
/***********************************************************************
Packit Service 5195f2
// this is a public function, but there is no prototype in a header file
Packit Service 5195f2
************************************************************************/
Packit Service 5195f2
Packit Service 5195f2
LZO_EXTERN(int)
Packit Service 5195f2
lzo2a_999_compress_callback ( 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
                                    lzo_callback_p cb,
Packit Service 5195f2
                                    lzo_uint max_chain );
Packit Service 5195f2
Packit Service 5195f2
LZO_PUBLIC(int)
Packit Service 5195f2
lzo2a_999_compress_callback ( 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
                                    lzo_callback_p cb,
Packit Service 5195f2
                                    lzo_uint max_chain )
Packit Service 5195f2
{
Packit Service 5195f2
    lzo_bytep op;
Packit Service 5195f2
    lzo_bytep bitp = 0;
Packit Service 5195f2
    lzo_uint m_len, m_off;
Packit Service 5195f2
    LZO_COMPRESS_T cc;
Packit Service 5195f2
    LZO_COMPRESS_T * const c = &cc;
Packit Service 5195f2
    lzo_swd_p const swd = (lzo_swd_p) wrkmem;
Packit Service 5195f2
    int r;
Packit Service 5195f2
Packit Service 5195f2
    lzo_uint32_t b = 0;     /* bit buffer */
Packit Service 5195f2
    unsigned k = 0;         /* bits in bit buffer */
Packit Service 5195f2
Packit Service 5195f2
    /* sanity check */
Packit Service 5195f2
    LZO_COMPILE_TIME_ASSERT(LZO2A_999_MEM_COMPRESS >= SIZEOF_LZO_SWD_T)
Packit Service 5195f2
Packit Service 5195f2
    c->init = 0;
Packit Service 5195f2
    c->ip = c->in = in;
Packit Service 5195f2
    c->in_end = in + in_len;
Packit Service 5195f2
    c->cb = cb;
Packit Service 5195f2
    c->m1 = c->m2 = c->m3 = c->m4 = 0;
Packit Service 5195f2
Packit Service 5195f2
    op = out;
Packit Service 5195f2
Packit Service 5195f2
    r = init_match(c,swd,NULL,0,0);
Packit Service 5195f2
    if (r != 0)
Packit Service 5195f2
        return r;
Packit Service 5195f2
    if (max_chain > 0)
Packit Service 5195f2
        swd->max_chain = max_chain;
Packit Service 5195f2
Packit Service 5195f2
    r = find_match(c,swd,0,0);
Packit Service 5195f2
    if (r != 0)
Packit Service 5195f2
        return r;
Packit Service 5195f2
    while (c->look > 0)
Packit Service 5195f2
    {
Packit Service 5195f2
        lzo_uint lazy_match_min_gain = 0;
Packit Service 5195f2
#if (SWD_N >= 8192)
Packit Service 5195f2
        lzo_uint extra1 = 0;
Packit Service 5195f2
#endif
Packit Service 5195f2
        lzo_uint extra2 = 0;
Packit Service 5195f2
        lzo_uint ahead = 0;
Packit Service 5195f2
Packit Service 5195f2
        m_len = c->m_len;
Packit Service 5195f2
        m_off = c->m_off;
Packit Service 5195f2
Packit Service 5195f2
#if (SWD_N >= 8192)
Packit Service 5195f2
        if (m_off >= 8192)
Packit Service 5195f2
        {
Packit Service 5195f2
            if (m_len < M3_MIN_LEN)
Packit Service 5195f2
                m_len = 0;
Packit Service 5195f2
            else
Packit Service 5195f2
                lazy_match_min_gain = 1;
Packit Service 5195f2
        }
Packit Service 5195f2
        else
Packit Service 5195f2
#endif
Packit Service 5195f2
        if (m_len >= M1_MIN_LEN && m_len <= M1_MAX_LEN && m_off <= 256)
Packit Service 5195f2
        {
Packit Service 5195f2
            lazy_match_min_gain = 2;
Packit Service 5195f2
#if (SWD_N >= 8192)
Packit Service 5195f2
            extra1 = 3;
Packit Service 5195f2
#endif
Packit Service 5195f2
            extra2 = 2;
Packit Service 5195f2
        }
Packit Service 5195f2
        else if (m_len >= 10)
Packit Service 5195f2
            lazy_match_min_gain = 1;
Packit Service 5195f2
        else if (m_len >= 3)
Packit Service 5195f2
        {
Packit Service 5195f2
            lazy_match_min_gain = 1;
Packit Service 5195f2
#if (SWD_N >= 8192)
Packit Service 5195f2
            extra1 = 1;
Packit Service 5195f2
#endif
Packit Service 5195f2
        }
Packit Service 5195f2
        else
Packit Service 5195f2
            m_len = 0;
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
        /* try a lazy match */
Packit Service 5195f2
        if (lazy_match_min_gain > 0 && c->look > m_len)
Packit Service 5195f2
        {
Packit Service 5195f2
            unsigned char lit = LZO_BYTE(swd->b_char);
Packit Service 5195f2
Packit Service 5195f2
            r = find_match(c,swd,1,0);
Packit Service 5195f2
            assert(r == 0); LZO_UNUSED(r);
Packit Service 5195f2
            assert(c->look > 0);
Packit Service 5195f2
Packit Service 5195f2
#if (SWD_N >= 8192)
Packit Service 5195f2
            if (m_off < 8192 && c->m_off >= 8192)
Packit Service 5195f2
                lazy_match_min_gain += extra1;
Packit Service 5195f2
            else
Packit Service 5195f2
#endif
Packit Service 5195f2
            if (m_len >= M1_MIN_LEN && m_len <= M1_MAX_LEN && m_off <= 256)
Packit Service 5195f2
            {
Packit Service 5195f2
                if (!(c->m_len >= M1_MIN_LEN &&
Packit Service 5195f2
                      c->m_len <= M1_MAX_LEN && c->m_off <= 256))
Packit Service 5195f2
                    lazy_match_min_gain += extra2;
Packit Service 5195f2
            }
Packit Service 5195f2
            if (c->m_len >= M1_MIN_LEN &&
Packit Service 5195f2
                c->m_len <= M1_MAX_LEN && c->m_off <= 256)
Packit Service 5195f2
            {
Packit Service 5195f2
                    lazy_match_min_gain -= 1;
Packit Service 5195f2
            }
Packit Service 5195f2
Packit Service 5195f2
            if ((lzo_int) lazy_match_min_gain < 1)
Packit Service 5195f2
                lazy_match_min_gain = 1;
Packit Service 5195f2
Packit Service 5195f2
            if (c->m_len >= m_len + lazy_match_min_gain)
Packit Service 5195f2
            {
Packit Service 5195f2
                c->lazy++;
Packit Service 5195f2
#if !defined(NDEBUG)
Packit Service 5195f2
                m_len = c->m_len;
Packit Service 5195f2
                m_off = c->m_off;
Packit Service 5195f2
                assert(lzo_memcmp(c->ip - c->look, c->ip - c->look - m_off,
Packit Service 5195f2
                                  m_len) == 0);
Packit Service 5195f2
                assert(m_len >= 3 || (m_len >= 2 && m_off <= 256));
Packit Service 5195f2
#endif
Packit Service 5195f2
                /* code literal */
Packit Service 5195f2
                putbit(0);
Packit Service 5195f2
                putbyte(lit);
Packit Service 5195f2
                c->lit_bytes++;
Packit Service 5195f2
                continue;
Packit Service 5195f2
            }
Packit Service 5195f2
            else
Packit Service 5195f2
                ahead = 1;
Packit Service 5195f2
            assert(m_len > 0);
Packit Service 5195f2
        }
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
        if (m_len == 0)
Packit Service 5195f2
        {
Packit Service 5195f2
            /* a literal */
Packit Service 5195f2
            putbit(0);
Packit Service 5195f2
            putbyte(swd->b_char);
Packit Service 5195f2
            c->lit_bytes++;
Packit Service 5195f2
            r = find_match(c,swd,1,0);
Packit Service 5195f2
            assert(r == 0); LZO_UNUSED(r);
Packit Service 5195f2
        }
Packit Service 5195f2
        else
Packit Service 5195f2
        {
Packit Service 5195f2
            assert(m_len >= M1_MIN_LEN);
Packit Service 5195f2
            assert(m_off > 0);
Packit Service 5195f2
            assert(m_off <= SWD_N);
Packit Service 5195f2
Packit Service 5195f2
            /* 2 - code match */
Packit Service 5195f2
            if (m_len >= M1_MIN_LEN && m_len <= M1_MAX_LEN && m_off <= 256)
Packit Service 5195f2
            {
Packit Service 5195f2
                putbit(1);
Packit Service 5195f2
                putbit(0);
Packit Service 5195f2
                putbits(2,m_len - M1_MIN_LEN);
Packit Service 5195f2
                putbyte(m_off - 1);
Packit Service 5195f2
                c->m1++;
Packit Service 5195f2
            }
Packit Service 5195f2
#if (SWD_N >= 8192)
Packit Service 5195f2
            else if (m_off >= 8192)
Packit Service 5195f2
            {
Packit Service 5195f2
                unsigned len = m_len;
Packit Service 5195f2
                assert(m_len >= M3_MIN_LEN);
Packit Service 5195f2
                putbit(1);
Packit Service 5195f2
                putbit(1);
Packit Service 5195f2
                putbyte(m_off & 31);
Packit Service 5195f2
                putbyte(m_off >> 5);
Packit Service 5195f2
                putbit(1);
Packit Service 5195f2
                len -= M3_MIN_LEN - 1;
Packit Service 5195f2
                while (len > 255)
Packit Service 5195f2
                {
Packit Service 5195f2
                    len -= 255;
Packit Service 5195f2
                    putbyte(0);
Packit Service 5195f2
                }
Packit Service 5195f2
                putbyte(len);
Packit Service 5195f2
                c->m4++;
Packit Service 5195f2
            }
Packit Service 5195f2
#endif
Packit Service 5195f2
            else
Packit Service 5195f2
            {
Packit Service 5195f2
                assert(m_len >= 3);
Packit Service 5195f2
Packit Service 5195f2
                putbit(1);
Packit Service 5195f2
                putbit(1);
Packit Service 5195f2
                if (m_len <= 9)
Packit Service 5195f2
                {
Packit Service 5195f2
                    putbyte(((m_len - 2) << 5) | (m_off & 31));
Packit Service 5195f2
                    putbyte(m_off >> 5);
Packit Service 5195f2
                    c->m2++;
Packit Service 5195f2
                }
Packit Service 5195f2
                else
Packit Service 5195f2
                {
Packit Service 5195f2
                    lzo_uint len = m_len;
Packit Service 5195f2
                    putbyte(m_off & 31);
Packit Service 5195f2
                    putbyte(m_off >> 5);
Packit Service 5195f2
#if (SWD_N >= 8192)
Packit Service 5195f2
                    putbit(0);
Packit Service 5195f2
#endif
Packit Service 5195f2
                    len -= 10 - 1;
Packit Service 5195f2
                    while (len > 255)
Packit Service 5195f2
                    {
Packit Service 5195f2
                        len -= 255;
Packit Service 5195f2
                        putbyte(0);
Packit Service 5195f2
                    }
Packit Service 5195f2
                    putbyte(len);
Packit Service 5195f2
                    c->m3++;
Packit Service 5195f2
                }
Packit Service 5195f2
            }
Packit Service 5195f2
            r = find_match(c,swd,m_len,1+ahead);
Packit Service 5195f2
            assert(r == 0); LZO_UNUSED(r);
Packit Service 5195f2
        }
Packit Service 5195f2
Packit Service 5195f2
        c->codesize = pd(op, out);
Packit Service 5195f2
    }
Packit Service 5195f2
Packit Service 5195f2
#if defined(LZO_EOF_CODE)
Packit Service 5195f2
    /* code EOF code */
Packit Service 5195f2
    putbit(1);
Packit Service 5195f2
    putbit(1);
Packit Service 5195f2
    putbyte(1 << 5);
Packit Service 5195f2
    putbyte(0);
Packit Service 5195f2
#endif
Packit Service 5195f2
Packit Service 5195f2
    /* flush remaining bits */
Packit Service 5195f2
    assert(k < CHAR_BIT);
Packit Service 5195f2
    if (k > 0)
Packit Service 5195f2
    {
Packit Service 5195f2
        assert(b == MASKBITS(k));
Packit Service 5195f2
        assert(op - bitp > 1);
Packit Service 5195f2
        *bitp = LZO_BYTE(MASKBITS(k));
Packit Service 5195f2
        DUMPBITS(k);
Packit Service 5195f2
        assert(b == 0);
Packit Service 5195f2
        assert(k == 0);
Packit Service 5195f2
    }
Packit Service 5195f2
Packit Service 5195f2
    assert(c->textsize == in_len);
Packit Service 5195f2
    c->codesize = pd(op, out);
Packit Service 5195f2
Packit Service 5195f2
    *out_len = pd(op, out);
Packit Service 5195f2
Packit Service 5195f2
    if (c->cb && c->cb->nprogress)
Packit Service 5195f2
        (*c->cb->nprogress)(c->cb, c->textsize, c->codesize, 0);
Packit Service 5195f2
Packit Service 5195f2
#if 0
Packit Service 5195f2
    printf("%ld -> %ld: %ld %ld %ld %ld %ld %ld\n",
Packit Service 5195f2
        (long) c->textsize, (long) c->codesize,
Packit Service 5195f2
        c->lit_bytes, c->m1, c->m2, c->m3, c->m4, c->lazy);
Packit Service 5195f2
#endif
Packit Service 5195f2
    return LZO_E_OK;
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
LZO_PUBLIC(int)
Packit Service 5195f2
lzo2a_999_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
    return lzo2a_999_compress_callback(in,in_len,out,out_len,wrkmem,
Packit Service 5195f2
                                       (lzo_callback_p) 0, 0);
Packit Service 5195f2
}
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
/*
Packit Service 5195f2
vi:ts=4:et
Packit Service 5195f2
*/
Packit Service 5195f2