Blame src/lzo1f_1.c

Packit Service 5195f2
/* lzo1f_1.c -- implementation of the LZO1F-1 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
#include "lzo_conf.h"
Packit Service 5195f2
#include "lzo/lzo1f.h"
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
/***********************************************************************
Packit Service 5195f2
//
Packit Service 5195f2
************************************************************************/
Packit Service 5195f2
Packit Service 5195f2
#define M2_MAX_OFFSET   0x0800
Packit Service 5195f2
#define M3_MAX_OFFSET   0x3fff
Packit Service 5195f2
#define M3_MARKER       224
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
#ifndef LZO_HASH
Packit Service 5195f2
#define LZO_HASH        LZO_HASH_LZO_INCREMENTAL_A
Packit Service 5195f2
#endif
Packit Service 5195f2
#define D_BITS          14
Packit Service 5195f2
#define D_INDEX1(d,p)   d = DM(DMUL(0x21,DX3(p,5,5,6)) >> 5)
Packit Service 5195f2
#define D_INDEX2(d,p)   d = (d & (D_MASK & 0x7ff)) ^ (D_HIGH | 0x1f)
Packit Service 5195f2
#include "lzo_dict.h"
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
/***********************************************************************
Packit Service 5195f2
// compress a block of data.
Packit Service 5195f2
************************************************************************/
Packit Service 5195f2
Packit Service 5195f2
static __lzo_noinline
Packit Service 5195f2
int 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
    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 - 9;
Packit Service 5195f2
    const lzo_bytep ii;
Packit Service 5195f2
    lzo_dict_p const dict = (lzo_dict_p) wrkmem;
Packit Service 5195f2
Packit Service 5195f2
    op = out;
Packit Service 5195f2
    ip = in;
Packit Service 5195f2
    ii = ip;
Packit Service 5195f2
Packit Service 5195f2
    ip++;
Packit Service 5195f2
    for (;;)
Packit Service 5195f2
    {
Packit Service 5195f2
        const lzo_bytep m_pos;
Packit Service 5195f2
        LZO_DEFINE_UNINITIALIZED_VAR(lzo_uint, m_off, 0);
Packit Service 5195f2
        lzo_uint m_len;
Packit Service 5195f2
        lzo_uint dindex;
Packit Service 5195f2
        lzo_uint lit;
Packit Service 5195f2
Packit Service 5195f2
        DINDEX1(dindex,ip);
Packit Service 5195f2
        GINDEX(m_pos,m_off,dict,dindex,in);
Packit Service 5195f2
        if (LZO_CHECK_MPOS_NON_DET(m_pos,m_off,in,ip,M3_MAX_OFFSET))
Packit Service 5195f2
            goto literal;
Packit Service 5195f2
#if 1
Packit Service 5195f2
        if (m_off <= M2_MAX_OFFSET || m_pos[3] == ip[3])
Packit Service 5195f2
            goto try_match;
Packit Service 5195f2
        DINDEX2(dindex,ip);
Packit Service 5195f2
#endif
Packit Service 5195f2
        GINDEX(m_pos,m_off,dict,dindex,in);
Packit Service 5195f2
        if (LZO_CHECK_MPOS_NON_DET(m_pos,m_off,in,ip,M3_MAX_OFFSET))
Packit Service 5195f2
            goto literal;
Packit Service 5195f2
        if (m_off <= M2_MAX_OFFSET || m_pos[3] == ip[3])
Packit Service 5195f2
            goto try_match;
Packit Service 5195f2
        goto literal;
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
try_match:
Packit Service 5195f2
#if 0 && (LZO_OPT_UNALIGNED16)
Packit Service 5195f2
        if (UA_GET_NE16(m_pos) != UA_GET_NE16(ip))
Packit Service 5195f2
#else
Packit Service 5195f2
        if (m_pos[0] != ip[0] || m_pos[1] != ip[1])
Packit Service 5195f2
#endif
Packit Service 5195f2
        {
Packit Service 5195f2
        }
Packit Service 5195f2
        else
Packit Service 5195f2
        {
Packit Service 5195f2
            if (m_pos[2] == ip[2])
Packit Service 5195f2
            {
Packit Service 5195f2
                m_pos += 3;
Packit Service 5195f2
#if 0
Packit Service 5195f2
                if (m_off <= M2_MAX_OFFSET)
Packit Service 5195f2
                    goto match;
Packit Service 5195f2
                if (lit <= 3)
Packit Service 5195f2
                    goto match;
Packit Service 5195f2
                if (lit == 3)           /* better compression, but slower */
Packit Service 5195f2
                {
Packit Service 5195f2
                    assert(op - 2 > out); op[-2] |= LZO_BYTE(3);
Packit Service 5195f2
                    *op++ = *ii++; *op++ = *ii++; *op++ = *ii++;
Packit Service 5195f2
                    goto code_match;
Packit Service 5195f2
                }
Packit Service 5195f2
                if (*m_pos == ip[3])
Packit Service 5195f2
#endif
Packit Service 5195f2
                    goto match;
Packit Service 5195f2
            }
Packit Service 5195f2
        }
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
    /* a literal */
Packit Service 5195f2
literal:
Packit Service 5195f2
        UPDATE_I(dict,0,dindex,ip,in);
Packit Service 5195f2
        if (++ip >= ip_end)
Packit Service 5195f2
            break;
Packit Service 5195f2
        continue;
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
    /* a match */
Packit Service 5195f2
match:
Packit Service 5195f2
        UPDATE_I(dict,0,dindex,ip,in);
Packit Service 5195f2
        /* store current literal run */
Packit Service 5195f2
        lit = pd(ip,ii);
Packit Service 5195f2
        if (lit > 0)
Packit Service 5195f2
        {
Packit Service 5195f2
            lzo_uint t = lit;
Packit Service 5195f2
Packit Service 5195f2
            if (t < 4 && op > out)
Packit Service 5195f2
                op[-2] = LZO_BYTE(op[-2] | t);
Packit Service 5195f2
            else if (t <= 31)
Packit Service 5195f2
                *op++ = LZO_BYTE(t);
Packit Service 5195f2
            else
Packit Service 5195f2
            {
Packit Service 5195f2
                lzo_uint tt = t - 31;
Packit Service 5195f2
Packit Service 5195f2
                *op++ = 0;
Packit Service 5195f2
                while (tt > 255)
Packit Service 5195f2
                {
Packit Service 5195f2
                    tt -= 255;
Packit Service 5195f2
                    UA_SET1(op, 0);
Packit Service 5195f2
                    op++;
Packit Service 5195f2
                }
Packit Service 5195f2
                assert(tt > 0);
Packit Service 5195f2
                *op++ = LZO_BYTE(tt);
Packit Service 5195f2
            }
Packit Service 5195f2
            do *op++ = *ii++; while (--t > 0);
Packit Service 5195f2
        }
Packit Service 5195f2
        assert(ii == ip);
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
        /* code the match */
Packit Service 5195f2
        ip += 3;
Packit Service 5195f2
        if (*m_pos++ != *ip++ || *m_pos++ != *ip++ || *m_pos++ != *ip++ ||
Packit Service 5195f2
            *m_pos++ != *ip++ || *m_pos++ != *ip++ || *m_pos++ != *ip++)
Packit Service 5195f2
        {
Packit Service 5195f2
            --ip;
Packit Service 5195f2
            m_len = pd(ip, ii);
Packit Service 5195f2
            assert(m_len >= 3); assert(m_len <= 8);
Packit Service 5195f2
Packit Service 5195f2
            if (m_off <= M2_MAX_OFFSET)
Packit Service 5195f2
            {
Packit Service 5195f2
                m_off -= 1;
Packit Service 5195f2
                *op++ = LZO_BYTE(((m_len - 2) << 5) | ((m_off & 7) << 2));
Packit Service 5195f2
                *op++ = LZO_BYTE(m_off >> 3);
Packit Service 5195f2
            }
Packit Service 5195f2
            else if (m_len == 3 && m_off <= 2*M2_MAX_OFFSET && lit > 0)
Packit Service 5195f2
            {
Packit Service 5195f2
                m_off -= 1;
Packit Service 5195f2
                /* m_off -= M2_MAX_OFFSET; */
Packit Service 5195f2
                *op++ = LZO_BYTE(((m_off & 7) << 2));
Packit Service 5195f2
                *op++ = LZO_BYTE(m_off >> 3);
Packit Service 5195f2
            }
Packit Service 5195f2
            else
Packit Service 5195f2
            {
Packit Service 5195f2
                *op++ = LZO_BYTE(M3_MARKER | (m_len - 2));
Packit Service 5195f2
                *op++ = LZO_BYTE((m_off & 63) << 2);
Packit Service 5195f2
                *op++ = LZO_BYTE(m_off >> 6);
Packit Service 5195f2
            }
Packit Service 5195f2
        }
Packit Service 5195f2
        else
Packit Service 5195f2
        {
Packit Service 5195f2
            {
Packit Service 5195f2
                const lzo_bytep end;
Packit Service 5195f2
                end = in_end;
Packit Service 5195f2
                while (ip < end && *m_pos == *ip)
Packit Service 5195f2
                    m_pos++, ip++;
Packit Service 5195f2
                m_len = pd(ip, ii);
Packit Service 5195f2
            }
Packit Service 5195f2
            assert(m_len >= 3);
Packit Service 5195f2
Packit Service 5195f2
            if (m_len <= 33)
Packit Service 5195f2
                *op++ = LZO_BYTE(M3_MARKER | (m_len - 2));
Packit Service 5195f2
            else
Packit Service 5195f2
            {
Packit Service 5195f2
                m_len -= 33;
Packit Service 5195f2
                *op++ = M3_MARKER | 0;
Packit Service 5195f2
                while (m_len > 255)
Packit Service 5195f2
                {
Packit Service 5195f2
                    m_len -= 255;
Packit Service 5195f2
                    UA_SET1(op, 0);
Packit Service 5195f2
                    op++;
Packit Service 5195f2
                }
Packit Service 5195f2
                assert(m_len > 0);
Packit Service 5195f2
                *op++ = LZO_BYTE(m_len);
Packit Service 5195f2
            }
Packit Service 5195f2
            *op++ = LZO_BYTE((m_off & 63) << 2);
Packit Service 5195f2
            *op++ = LZO_BYTE(m_off >> 6);
Packit Service 5195f2
        }
Packit Service 5195f2
Packit Service 5195f2
        ii = ip;
Packit Service 5195f2
        if (ip >= ip_end)
Packit Service 5195f2
            break;
Packit Service 5195f2
    }
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
    /* store 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
Packit Service 5195f2
        if (t < 4 && op > out)
Packit Service 5195f2
            op[-2] = LZO_BYTE(op[-2] | t);
Packit Service 5195f2
        else if (t <= 31)
Packit Service 5195f2
            *op++ = LZO_BYTE(t);
Packit Service 5195f2
        else
Packit Service 5195f2
        {
Packit Service 5195f2
            lzo_uint tt = t - 31;
Packit Service 5195f2
Packit Service 5195f2
            *op++ = 0;
Packit Service 5195f2
            while (tt > 255)
Packit Service 5195f2
            {
Packit Service 5195f2
                tt -= 255;
Packit Service 5195f2
                UA_SET1(op, 0);
Packit Service 5195f2
                op++;
Packit Service 5195f2
            }
Packit Service 5195f2
            assert(tt > 0);
Packit Service 5195f2
            *op++ = LZO_BYTE(tt);
Packit Service 5195f2
        }
Packit Service 5195f2
        UA_COPYN(op, ii, t);
Packit Service 5195f2
        op += t;
Packit Service 5195f2
    }
Packit Service 5195f2
Packit Service 5195f2
    *out_len = pd(op, out);
Packit Service 5195f2
    return LZO_E_OK;
Packit Service 5195f2
}
Packit Service 5195f2
Packit Service 5195f2
Packit Service 5195f2
/***********************************************************************
Packit Service 5195f2
// public entry point
Packit Service 5195f2
************************************************************************/
Packit Service 5195f2
Packit Service 5195f2
LZO_PUBLIC(int)
Packit Service 5195f2
lzo1f_1_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
    lzo_bytep op = out;
Packit Service 5195f2
    int r = LZO_E_OK;
Packit Service 5195f2
Packit Service 5195f2
    if (in_len == 0)
Packit Service 5195f2
        *out_len = 0;
Packit Service 5195f2
    else if (in_len <= 10)
Packit Service 5195f2
    {
Packit Service 5195f2
        *op++ = LZO_BYTE(in_len);
Packit Service 5195f2
        do *op++ = *in++; while (--in_len > 0);
Packit Service 5195f2
        *out_len = pd(op, out);
Packit Service 5195f2
    }
Packit Service 5195f2
    else
Packit Service 5195f2
        r = do_compress(in,in_len,out,out_len,wrkmem);
Packit Service 5195f2
Packit Service 5195f2
    if (r == LZO_E_OK)
Packit Service 5195f2
    {
Packit Service 5195f2
        op = out + *out_len;
Packit Service 5195f2
        op[0] = M3_MARKER | 1;
Packit Service 5195f2
        op[1] = 0;
Packit Service 5195f2
        op[2] = 0;
Packit Service 5195f2
        *out_len += 3;
Packit Service 5195f2
    }
Packit Service 5195f2
Packit Service 5195f2
    return r;
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