Blame src/lzo1b_sm.ch

Packit 679830
/* lzo1b_sm.ch -- implementation of the LZO1B 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
// search for a match
Packit 679830
************************************************************************/
Packit 679830
Packit 679830
#if (DD_BITS == 0)
Packit 679830
Packit 679830
        /* search ip in the dictionary */
Packit 679830
        DINDEX1(dindex,ip);
Packit 679830
        GINDEX(m_pos,m_off,dict,dindex,in);
Packit 679830
        if (LZO_CHECK_MPOS_NON_DET(m_pos,m_off,in,ip,M3_MAX_OFFSET))
Packit 679830
            goto literal;
Packit 679830
#if 1
Packit 679830
        if (m_off <= M2_MAX_OFFSET || m_pos[3] == ip[3])
Packit 679830
            goto try_match;
Packit 679830
        DINDEX2(dindex,ip);
Packit 679830
#endif
Packit 679830
        GINDEX(m_pos,m_off,dict,dindex,in);
Packit 679830
        if (LZO_CHECK_MPOS_NON_DET(m_pos,m_off,in,ip,M3_MAX_OFFSET))
Packit 679830
            goto literal;
Packit 679830
        if (m_off <= M2_MAX_OFFSET || m_pos[3] == ip[3])
Packit 679830
            goto try_match;
Packit 679830
        goto literal;
Packit 679830
Packit 679830
Packit 679830
#else /* (DD_BITS == 0) */
Packit 679830
Packit 679830
Packit 679830
Packit 679830
        /* search ip in the deepened dictionary */
Packit 679830
        {
Packit 679830
            lzo_dict_p d = &dict [ DINDEX(dv,ip) ];
Packit 679830
            const lzo_bytep ip_sav;
Packit 679830
            unsigned j = DD_SIZE;
Packit 679830
            lzo_uint x_len;
Packit 679830
            LZO_DEFINE_UNINITIALIZED_VAR(lzo_uint, x_off, 0);
Packit 679830
Packit 679830
            DVAL_ASSERT(dv,ip);
Packit 679830
Packit 679830
            ip_sav = ip;
Packit 679830
            m_len = 0;
Packit 679830
            do {
Packit 679830
#if !defined(NDEBUG)
Packit 679830
                const lzo_bytep z_pos = NULL;
Packit 679830
#endif
Packit 679830
#if (LZO_DICT_USE_PTR)
Packit 679830
                m_pos = *d;
Packit 679830
                assert((z_pos = m_pos) == *d);
Packit 679830
#if (LZO_DETERMINISTIC)
Packit 679830
                assert(m_pos == NULL || m_pos >= in);
Packit 679830
                assert(m_pos == NULL || m_pos < ip);
Packit 679830
#endif
Packit 679830
#else
Packit 679830
                x_off = *d;
Packit 679830
#endif
Packit 679830
                assert(ip == ip_sav);
Packit 679830
Packit 679830
                if (LZO_CHECK_MPOS(m_pos,x_off,in,ip,MAX_OFFSET))
Packit 679830
#if (CLEVEL == 9)
Packit 679830
                    *d = DENTRY(ip,in);
Packit 679830
#else
Packit 679830
                    ((void)(0));
Packit 679830
#endif
Packit 679830
                else if (m_pos[m_len] != ip[m_len])
Packit 679830
                    ((void)(0));
Packit 679830
                else if (*m_pos++ == *ip++ && *m_pos++ == *ip++ && *m_pos++ == *ip++)
Packit 679830
                {
Packit 679830
#if !(LZO_DICT_USE_PTR)
Packit 679830
                    assert((z_pos = ip - 3 - x_off) == (m_pos - 3));
Packit 679830
#endif
Packit 679830
                    /* a match */
Packit 679830
                    if (MATCH_M2)
Packit 679830
                    {
Packit 679830
                        x_len = pd((ip - 1), ip_sav);
Packit 679830
                        if (x_len > m_len)
Packit 679830
                        {
Packit 679830
                            m_len = x_len;
Packit 679830
                            m_off = x_off;
Packit 679830
                            assert((m_pos_sav = z_pos) != NULL);
Packit 679830
                        }
Packit 679830
#if (CLEVEL == 9)
Packit 679830
                        /* try to find a closer match */
Packit 679830
                        else if (x_len == m_len && x_off < m_off)
Packit 679830
                        {
Packit 679830
                            m_off = x_off;
Packit 679830
                            assert((m_pos_sav = z_pos) != NULL);
Packit 679830
                        }
Packit 679830
#endif
Packit 679830
                    }
Packit 679830
                    else
Packit 679830
                    {
Packit 679830
                        assert((ip - ip_sav) == M2_MAX_LEN + 1);
Packit 679830
#if (CLEVEL == 9)
Packit 679830
#if defined(MATCH_IP_END)
Packit 679830
                        {
Packit 679830
                            const lzo_bytep end;
Packit 679830
                            end = MATCH_IP_END;
Packit 679830
                            while (ip < end  &&  *m_pos == *ip)
Packit 679830
                                m_pos++, ip++;
Packit 679830
                            assert(ip <= in_end);
Packit 679830
                            x_len = pd(ip, ip_sav);
Packit 679830
                        }
Packit 679830
                        if (x_len > m_len)
Packit 679830
                        {
Packit 679830
                            m_len = x_len;
Packit 679830
                            m_off = x_off;
Packit 679830
                            assert((m_pos_sav = z_pos) != NULL);
Packit 679830
                            if (ip >= MATCH_IP_END)
Packit 679830
                            {
Packit 679830
                                ip = ip_sav;
Packit 679830
#if 0
Packit 679830
                                /* not needed - we are at the end */
Packit 679830
                                d -= DD_SIZE - j;
Packit 679830
                                assert(d == &dict [ DINDEX(dv,ip) ]);
Packit 679830
                                UPDATE_P(d,drun,ip,in);
Packit 679830
#endif
Packit 679830
                                goto match;
Packit 679830
                            }
Packit 679830
                        }
Packit 679830
                        else if (x_len == m_len && x_off < m_off)
Packit 679830
                        {
Packit 679830
                            m_off = x_off;
Packit 679830
                            assert((m_pos_sav = z_pos) != NULL);
Packit 679830
                        }
Packit 679830
#else
Packit 679830
                        /* try to find a closer match */
Packit 679830
                        if (m_len < M2_MAX_LEN + 1 || x_off < m_off)
Packit 679830
                        {
Packit 679830
                            m_len = M2_MAX_LEN + 1;
Packit 679830
                            m_off = x_off;
Packit 679830
                            assert((m_pos_sav = z_pos) != NULL);
Packit 679830
                        }
Packit 679830
#endif
Packit 679830
#else
Packit 679830
                        /* don't search for a longer/closer match */
Packit 679830
                        m_len = M2_MAX_LEN + 1;
Packit 679830
                        m_off = x_off;
Packit 679830
                        assert((m_pos_sav = z_pos) != NULL);
Packit 679830
                        ip = ip_sav;
Packit 679830
                        d -= DD_SIZE - j;
Packit 679830
                        assert(d == &dict [ DINDEX(dv,ip) ]);
Packit 679830
                        UPDATE_P(d,drun,ip,in);
Packit 679830
                        goto match;
Packit 679830
#endif
Packit 679830
                    }
Packit 679830
                    ip = ip_sav;
Packit 679830
                }
Packit 679830
                else
Packit 679830
                    ip = ip_sav;
Packit 679830
                d++;
Packit 679830
            } while (--j > 0);
Packit 679830
            assert(ip == ip_sav);
Packit 679830
Packit 679830
            d -= DD_SIZE;
Packit 679830
            assert(d == &dict [ DINDEX(dv,ip) ]);
Packit 679830
            UPDATE_P(d,drun,ip,in);
Packit 679830
        }
Packit 679830
Packit 679830
#endif /* (DD_BITS == 0) */
Packit 679830
Packit 679830
Packit 679830
/*
Packit 679830
vi:ts=4:et
Packit 679830
*/