Blame src/lzo1b_d.ch

Packit 679830
/* lzo1b_d.ch -- implementation of the LZO1B decompression 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
#include "lzo1_d.ch"
Packit 679830
Packit 679830
Packit 679830
/***********************************************************************
Packit 679830
// decompress a block of data.
Packit 679830
************************************************************************/
Packit 679830
Packit 679830
LZO_PUBLIC(int)
Packit 679830
DO_DECOMPRESS    ( const lzo_bytep in , lzo_uint  in_len,
Packit 679830
                         lzo_bytep out, lzo_uintp out_len,
Packit 679830
                         lzo_voidp wrkmem )
Packit 679830
{
Packit 679830
    lzo_bytep op;
Packit 679830
    const lzo_bytep ip;
Packit 679830
    lzo_uint t;
Packit 679830
    const lzo_bytep m_pos;
Packit 679830
Packit 679830
    const lzo_bytep const ip_end = in + in_len;
Packit 679830
#if defined(HAVE_ANY_OP)
Packit 679830
    lzo_bytep const op_end = out + *out_len;
Packit 679830
#endif
Packit 679830
Packit 679830
    LZO_UNUSED(wrkmem);
Packit 679830
Packit 679830
    op = out;
Packit 679830
    ip = in;
Packit 679830
Packit 679830
    while (TEST_IP_AND_TEST_OP)
Packit 679830
    {
Packit 679830
        t = *ip++;      /* get marker */
Packit 679830
Packit 679830
        if (t < R0MIN)      /* a literal run */
Packit 679830
        {
Packit 679830
            if (t == 0)             /* a R0 literal run */
Packit 679830
            {
Packit 679830
                NEED_IP(1);
Packit 679830
                t = *ip++;
Packit 679830
                if (t >= R0FAST - R0MIN)            /* a long R0 run */
Packit 679830
                {
Packit 679830
                    t -= R0FAST - R0MIN;
Packit 679830
                    if (t == 0)
Packit 679830
                        t = R0FAST;
Packit 679830
                    else
Packit 679830
                    {
Packit 679830
#if 0
Packit 679830
                        t = 256u << ((unsigned) t);
Packit 679830
#else
Packit 679830
                        /* help the optimizer */
Packit 679830
                        lzo_uint tt = 256;
Packit 679830
                        do tt <<= 1; while (--t > 0);
Packit 679830
                        t = tt;
Packit 679830
#endif
Packit 679830
                    }
Packit 679830
Packit 679830
                    NEED_IP(t); NEED_OP(t);
Packit 679830
#if 1 && (LZO_OPT_UNALIGNED32)
Packit 679830
                    do {
Packit 679830
                        UA_COPY4(op+0, ip+0);
Packit 679830
                        UA_COPY4(op+4, ip+4);
Packit 679830
                        op += 8; ip += 8;
Packit 679830
                        t -= 8;
Packit 679830
                    } while (t > 0);
Packit 679830
#else
Packit 679830
                    MEMCPY8_DS(op,ip,t);
Packit 679830
#endif
Packit 679830
                    continue;
Packit 679830
                }
Packit 679830
                t += R0MIN;                         /* a short R0 run */
Packit 679830
            }
Packit 679830
Packit 679830
            NEED_IP(t); NEED_OP(t);
Packit 679830
            /* copy literal run */
Packit 679830
#if 1 && (LZO_OPT_UNALIGNED32)
Packit 679830
            if (t >= 4)
Packit 679830
            {
Packit 679830
                do {
Packit 679830
                    UA_COPY4(op, ip);
Packit 679830
                    op += 4; ip += 4; t -= 4;
Packit 679830
                } while (t >= 4);
Packit 679830
                if (t > 0) do *op++ = *ip++; while (--t > 0);
Packit 679830
            }
Packit 679830
            else
Packit 679830
#endif
Packit 679830
            {
Packit 679830
#if (M3O_BITS < 7)
Packit 679830
literal1:
Packit 679830
#endif
Packit 679830
                do *op++ = *ip++; while (--t > 0);
Packit 679830
            }
Packit 679830
Packit 679830
#if (M3O_BITS == 7)
Packit 679830
literal2:
Packit 679830
#endif
Packit 679830
Packit 679830
            /* after a literal a match must follow */
Packit 679830
            while (TEST_IP_AND_TEST_OP)
Packit 679830
            {
Packit 679830
                t = *ip++;          /* get R1 marker */
Packit 679830
                if (t >= R0MIN)
Packit 679830
                    goto match;
Packit 679830
Packit 679830
                NEED_IP(2); NEED_OP(M2_MIN_LEN + 1);
Packit 679830
Packit 679830
            /* R1 match - a M2_MIN_LEN match + 1 byte literal */
Packit 679830
                assert((t & M2O_MASK) == t);
Packit 679830
                m_pos = op - M2_MIN_OFFSET;
Packit 679830
                m_pos -= t | (((lzo_uint) *ip++) << M2O_BITS);
Packit 679830
                assert(m_pos >= out); assert(m_pos < op);
Packit 679830
                TEST_LB(m_pos);
Packit 679830
                COPY_M2;
Packit 679830
                *op++ = *ip++;
Packit 679830
            }
Packit 679830
Packit 679830
#if defined(HAVE_TEST_IP) || defined(HAVE_TEST_OP)
Packit 679830
            break;
Packit 679830
#endif
Packit 679830
        }
Packit 679830
Packit 679830
match:
Packit 679830
Packit 679830
        if (t >= M2_MARKER)             /* a M2 match */
Packit 679830
        {
Packit 679830
            /* get match offset */
Packit 679830
            NEED_IP(1);
Packit 679830
            m_pos = op - M2_MIN_OFFSET;
Packit 679830
            m_pos -= (t & M2O_MASK) | (((lzo_uint) *ip++) << M2O_BITS);
Packit 679830
            assert(m_pos >= out); assert(m_pos < op);
Packit 679830
            TEST_LB(m_pos);
Packit 679830
Packit 679830
            /* get match len */
Packit 679830
            t = (t >> M2O_BITS) - 1;
Packit 679830
            NEED_OP(t + M2_MIN_LEN - 1);
Packit 679830
            COPY_M2X;
Packit 679830
            MEMCPY_DS(op,m_pos,t);
Packit 679830
        }
Packit 679830
        else                            /* a M3 or M4 match */
Packit 679830
        {
Packit 679830
            /* get match len */
Packit 679830
            t &= M3L_MASK;
Packit 679830
            if (t == 0)         /* a M4 match */
Packit 679830
            {
Packit 679830
                NEED_IP(1);
Packit 679830
                while (*ip == 0)
Packit 679830
                {
Packit 679830
                    t += 255;
Packit 679830
                    ip++;
Packit 679830
                    TEST_OV(t);
Packit 679830
                    NEED_IP(1);
Packit 679830
                }
Packit 679830
                t += (M4_MIN_LEN - M3_MIN_LEN) + *ip++;
Packit 679830
            }
Packit 679830
Packit 679830
            /* get match offset */
Packit 679830
            NEED_IP(2);
Packit 679830
            m_pos = op - (M3_MIN_OFFSET - M3_EOF_OFFSET);
Packit 679830
            m_pos -= *ip++ & M3O_MASK;
Packit 679830
            m_pos -= (lzo_uint)(*ip++) << M3O_BITS;
Packit 679830
#if defined(LZO_EOF_CODE)
Packit 679830
            if (m_pos == op)
Packit 679830
                goto eof_found;
Packit 679830
#endif
Packit 679830
Packit 679830
            /* copy match */
Packit 679830
            assert(m_pos >= out); assert(m_pos < op);
Packit 679830
            TEST_LB(m_pos); NEED_OP(t + M3_MIN_LEN - 1);
Packit 679830
#if (LZO_OPT_UNALIGNED32)
Packit 679830
            if (t >= 2 * 4 - (M3_MIN_LEN - 1) && (op - m_pos) >= 4)
Packit 679830
            {
Packit 679830
                UA_COPY4(op, m_pos);
Packit 679830
                op += 4; m_pos += 4; t -= 4 - (M3_MIN_LEN - 1);
Packit 679830
                do {
Packit 679830
                    UA_COPY4(op, m_pos);
Packit 679830
                    op += 4; m_pos += 4; t -= 4;
Packit 679830
                } while (t >= 4);
Packit 679830
                if (t > 0) do *op++ = *m_pos++; while (--t > 0);
Packit 679830
            }
Packit 679830
            else
Packit 679830
#endif
Packit 679830
            {
Packit 679830
            COPY_M3X;
Packit 679830
            MEMCPY_DS(op,m_pos,t);
Packit 679830
            }
Packit 679830
Packit 679830
Packit 679830
#if (M3O_BITS < 7)
Packit 679830
            t = ip[-2] >> M3O_BITS;
Packit 679830
            if (t)
Packit 679830
            {
Packit 679830
                NEED_IP(t); NEED_OP(t);
Packit 679830
                goto literal1;
Packit 679830
            }
Packit 679830
#elif (M3O_BITS == 7)
Packit 679830
            /* optimized version */
Packit 679830
            if (ip[-2] & (1 << M3O_BITS))
Packit 679830
            {
Packit 679830
                NEED_IP(1); NEED_OP(1);
Packit 679830
                *op++ = *ip++;
Packit 679830
                goto literal2;
Packit 679830
            }
Packit 679830
#endif
Packit 679830
        }
Packit 679830
    }
Packit 679830
Packit 679830
Packit 679830
#if defined(LZO_EOF_CODE)
Packit 679830
#if defined(HAVE_TEST_IP) || defined(HAVE_TEST_OP)
Packit 679830
    /* no EOF code was found */
Packit 679830
    *out_len = pd(op, out);
Packit 679830
    return LZO_E_EOF_NOT_FOUND;
Packit 679830
#endif
Packit 679830
Packit 679830
eof_found:
Packit 679830
    assert(t == 1);
Packit 679830
#endif
Packit 679830
    *out_len = pd(op, out);
Packit 679830
    return (ip == ip_end ? LZO_E_OK :
Packit 679830
           (ip < ip_end  ? LZO_E_INPUT_NOT_CONSUMED : LZO_E_INPUT_OVERRUN));
Packit 679830
Packit 679830
Packit 679830
#if defined(HAVE_NEED_IP)
Packit 679830
input_overrun:
Packit 679830
    *out_len = pd(op, out);
Packit 679830
    return LZO_E_INPUT_OVERRUN;
Packit 679830
#endif
Packit 679830
Packit 679830
#if defined(HAVE_NEED_OP)
Packit 679830
output_overrun:
Packit 679830
    *out_len = pd(op, out);
Packit 679830
    return LZO_E_OUTPUT_OVERRUN;
Packit 679830
#endif
Packit 679830
Packit 679830
#if defined(LZO_TEST_OVERRUN_LOOKBEHIND)
Packit 679830
lookbehind_overrun:
Packit 679830
    *out_len = pd(op, out);
Packit 679830
    return LZO_E_LOOKBEHIND_OVERRUN;
Packit 679830
#endif
Packit 679830
}
Packit 679830
Packit 679830
Packit 679830
/*
Packit 679830
vi:ts=4:et
Packit 679830
*/
Packit 679830