Blame src/lzo1a_cm.ch

Packit 679830
/* lzo1a_cm.ch -- implementation of the LZO1A compression algorithm
Packit 679830
Packit 679830
   This file is part of the LZO real-time data compression library.
Packit 679830
Packit 679830
   Copyright (C) 1996-2014 Markus Franz Xaver Johannes Oberhumer
Packit 679830
   All Rights Reserved.
Packit 679830
Packit 679830
   The LZO library is free software; you can redistribute it and/or
Packit 679830
   modify it under the terms of the GNU General Public License as
Packit 679830
   published by the Free Software Foundation; either version 2 of
Packit 679830
   the License, or (at your option) any later version.
Packit 679830
Packit 679830
   The LZO library is distributed in the hope that it will be useful,
Packit 679830
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 679830
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 679830
   GNU General Public License for more details.
Packit 679830
Packit 679830
   You should have received a copy of the GNU General Public License
Packit 679830
   along with the LZO library; see the file COPYING.
Packit 679830
   If not, write to the Free Software Foundation, Inc.,
Packit 679830
   51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
Packit 679830
Packit 679830
   Markus F.X.J. Oberhumer
Packit 679830
   <markus@oberhumer.com>
Packit 679830
   http://www.oberhumer.com/opensource/lzo/
Packit 679830
 */
Packit 679830
Packit 679830
Packit 679830
/* WARNING: this file should *not* be used by applications. It is
Packit 679830
   part of the implementation of the library and is subject
Packit 679830
   to change.
Packit 679830
 */
Packit 679830
Packit 679830
Packit 679830
Packit 679830
/***********************************************************************
Packit 679830
// code the match in LZO1 compatible format
Packit 679830
************************************************************************/
Packit 679830
Packit 679830
#define THRESHOLD   (M2_MIN_LEN - 1)
Packit 679830
#define MSIZE       LZO_SIZE(M2L_BITS)
Packit 679830
Packit 679830
Packit 679830
/***********************************************************************
Packit 679830
//
Packit 679830
************************************************************************/
Packit 679830
Packit 679830
#if (DD_BITS == 0)
Packit 679830
Packit 679830
        /* we already matched M2_MIN_LEN bytes,
Packit 679830
         * m_pos also already advanced M2_MIN_LEN bytes */
Packit 679830
        ip += M2_MIN_LEN;
Packit 679830
        assert(m_pos < ip);
Packit 679830
Packit 679830
        /* try to match another M2_MAX_LEN + 1 - M2_MIN_LEN bytes
Packit 679830
         * to see if we get more than a M2 match */
Packit 679830
#define M2_OR_M3    (MATCH_M2)
Packit 679830
Packit 679830
#else /* (DD_BITS == 0) */
Packit 679830
Packit 679830
        /* we already matched m_len bytes */
Packit 679830
        assert(m_len >= M2_MIN_LEN);
Packit 679830
        ip += m_len;
Packit 679830
        assert(ip <= in_end);
Packit 679830
Packit 679830
#define M2_OR_M3    (m_len <= M2_MAX_LEN)
Packit 679830
Packit 679830
#endif /* (DD_BITS == 0) */
Packit 679830
Packit 679830
Packit 679830
        if (M2_OR_M3)
Packit 679830
        {
Packit 679830
        /* we've found a short match */
Packit 679830
            assert(ip <= in_end);
Packit 679830
Packit 679830
        /* 2a) compute match parameters */
Packit 679830
#if (DD_BITS == 0)
Packit 679830
                assert(pd(ip,m_pos) == m_off);
Packit 679830
            --ip;   /* ran one too far, point back to non-match */
Packit 679830
            m_len = ip - ii;
Packit 679830
#endif
Packit 679830
                assert(m_len >= M2_MIN_LEN);
Packit 679830
                assert(m_len <= M2_MAX_LEN);
Packit 679830
Packit 679830
                assert(m_off >= M2_MIN_OFFSET);
Packit 679830
                assert(m_off <= M2_MAX_OFFSET);
Packit 679830
                assert(ii-m_off == m_pos_sav);
Packit 679830
                assert(lzo_memcmp(m_pos_sav,ii,m_len) == 0);
Packit 679830
Packit 679830
        /* 2b) code the match */
Packit 679830
            m_off -= M2_MIN_OFFSET;
Packit 679830
            /* code short match len + low offset bits */
Packit 679830
            *op++ = LZO_BYTE(((m_len - THRESHOLD) << M2O_BITS) |
Packit 679830
                             (m_off & M2O_MASK));
Packit 679830
            /* code high offset bits */
Packit 679830
            *op++ = LZO_BYTE(m_off >> M2O_BITS);
Packit 679830
Packit 679830
Packit 679830
            if (ip >= ip_end)
Packit 679830
            {
Packit 679830
                ii = ip;
Packit 679830
                break;
Packit 679830
            }
Packit 679830
Packit 679830
Packit 679830
        /* 2c) Insert phrases (beginning with ii+1) into the dictionary. */
Packit 679830
Packit 679830
#if (CLEVEL == 9) || (CLEVEL >= 7 && M2L_BITS <= 4) || (CLEVEL >= 5 && M2L_BITS <= 3)
Packit 679830
        /* Insert the whole match (ii+1)..(ip-1) into dictionary.  */
Packit 679830
            ++ii;
Packit 679830
            do {
Packit 679830
                DVAL_NEXT(dv,ii);
Packit 679830
#if 0
Packit 679830
                UPDATE_D(dict,drun,dv,ii,in);
Packit 679830
#else
Packit 679830
                dict[ DINDEX(dv,ii) ] = DENTRY(ii,in);
Packit 679830
#endif
Packit 679830
                MI
Packit 679830
            } while (++ii < ip);
Packit 679830
            DVAL_NEXT(dv,ii);
Packit 679830
            assert(ii == ip);
Packit 679830
            DVAL_ASSERT(dv,ip);
Packit 679830
#elif (CLEVEL >= 3)
Packit 679830
            SI   DI DI   XI
Packit 679830
#elif (CLEVEL >= 2)
Packit 679830
            SI   DI      XI
Packit 679830
#else
Packit 679830
                         XI
Packit 679830
#endif
Packit 679830
        }
Packit 679830
Packit 679830
        else
Packit 679830
Packit 679830
        {
Packit 679830
        /* we've found a long match - see how far we can still go */
Packit 679830
            const lzo_bytep end;
Packit 679830
Packit 679830
            assert(ip <= in_end);
Packit 679830
            assert(ii == ip - (M2_MAX_LEN + 1));
Packit 679830
            assert(lzo_memcmp(m_pos_sav,ii,(lzo_uint)(ip-ii)) == 0);
Packit 679830
Packit 679830
#if (DD_BITS > 0)
Packit 679830
            assert(m_len == (lzo_uint)(ip-ii));
Packit 679830
            m_pos = ip - m_off;
Packit 679830
            assert(m_pos == m_pos_sav + m_len);
Packit 679830
#endif
Packit 679830
Packit 679830
            if (pd(in_end,ip) <= (M3_MAX_LEN - M3_MIN_LEN))
Packit 679830
                end = in_end;
Packit 679830
            else
Packit 679830
            {
Packit 679830
                end = ip + (M3_MAX_LEN - M3_MIN_LEN);
Packit 679830
                assert(end < in_end);
Packit 679830
            }
Packit 679830
Packit 679830
            while (ip < end  &&  *m_pos == *ip)
Packit 679830
                m_pos++, ip++;
Packit 679830
            assert(ip <= in_end);
Packit 679830
Packit 679830
            /* 2a) compute match parameters */
Packit 679830
            m_len = pd(ip, ii);
Packit 679830
                assert(m_len >= M3_MIN_LEN);
Packit 679830
                assert(m_len <= M3_MAX_LEN);
Packit 679830
Packit 679830
                assert(m_off >= M3_MIN_OFFSET);
Packit 679830
                assert(m_off <= M3_MAX_OFFSET);
Packit 679830
                assert(ii-m_off == m_pos_sav);
Packit 679830
                assert(lzo_memcmp(m_pos_sav,ii,m_len) == 0);
Packit 679830
                assert(pd(ip,m_pos) == m_off);
Packit 679830
Packit 679830
        /* 2b) code the match */
Packit 679830
            m_off -= M3_MIN_OFFSET - M3_EOF_OFFSET;
Packit 679830
            /* code long match flag + low offset bits */
Packit 679830
            *op++ = LZO_BYTE(((MSIZE - 1) << M3O_BITS) | (m_off & M3O_MASK));
Packit 679830
            /* code high offset bits */
Packit 679830
            *op++ = LZO_BYTE(m_off >> M3O_BITS);
Packit 679830
            /* code match len */
Packit 679830
            *op++ = LZO_BYTE(m_len - M3_MIN_LEN);
Packit 679830
Packit 679830
Packit 679830
            if (ip >= ip_end)
Packit 679830
            {
Packit 679830
                ii = ip;
Packit 679830
                break;
Packit 679830
            }
Packit 679830
Packit 679830
Packit 679830
        /* 2c) Insert phrases (beginning with ii+1) into the dictionary. */
Packit 679830
#if (CLEVEL == 9)
Packit 679830
        /* Insert the whole match (ii+1)..(ip-1) into dictionary.  */
Packit 679830
        /* This is not recommended because it can be slow. */
Packit 679830
            ++ii;
Packit 679830
            do {
Packit 679830
                DVAL_NEXT(dv,ii);
Packit 679830
#if 0
Packit 679830
                UPDATE_D(dict,drun,dv,ii,in);
Packit 679830
#else
Packit 679830
                dict[ DINDEX(dv,ii) ] = DENTRY(ii,in);
Packit 679830
#endif
Packit 679830
                MI
Packit 679830
            } while (++ii < ip);
Packit 679830
            DVAL_NEXT(dv,ii);
Packit 679830
            assert(ii == ip);
Packit 679830
            DVAL_ASSERT(dv,ip);
Packit 679830
#elif (CLEVEL >= 8)
Packit 679830
            SI   DI DI DI DI DI DI DI DI   XI
Packit 679830
#elif (CLEVEL >= 7)
Packit 679830
            SI   DI DI DI DI DI DI DI      XI
Packit 679830
#elif (CLEVEL >= 6)
Packit 679830
            SI   DI DI DI DI DI DI         XI
Packit 679830
#elif (CLEVEL >= 5)
Packit 679830
            SI   DI DI DI DI               XI
Packit 679830
#elif (CLEVEL >= 4)
Packit 679830
            SI   DI DI DI                  XI
Packit 679830
#elif (CLEVEL >= 3)
Packit 679830
            SI   DI DI                     XI
Packit 679830
#elif (CLEVEL >= 2)
Packit 679830
            SI   DI                        XI
Packit 679830
#else
Packit 679830
                                           XI
Packit 679830
#endif
Packit 679830
        }
Packit 679830
Packit 679830
        /* ii now points to the start of the next literal run */
Packit 679830
        assert(ii == ip);
Packit 679830
Packit 679830
Packit 679830
/*
Packit 679830
vi:ts=4:et
Packit 679830
*/