Blame src/lzo1x_d.ch

Packit 679830
/* lzo1x_d.ch -- implementation of the LZO1X 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
#if defined(DO_DECOMPRESS)
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
#endif
Packit 679830
{
Packit 679830
    lzo_bytep op;
Packit 679830
    const lzo_bytep ip;
Packit 679830
    lzo_uint t;
Packit 679830
#if defined(COPY_DICT)
Packit 679830
    lzo_uint m_off;
Packit 679830
    const lzo_bytep dict_end;
Packit 679830
#else
Packit 679830
    const lzo_bytep m_pos;
Packit 679830
#endif
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
#if defined(LZO1Z)
Packit 679830
    lzo_uint last_m_off = 0;
Packit 679830
#endif
Packit 679830
Packit 679830
    LZO_UNUSED(wrkmem);
Packit 679830
Packit 679830
#if defined(COPY_DICT)
Packit 679830
    if (dict)
Packit 679830
    {
Packit 679830
        if (dict_len > M4_MAX_OFFSET)
Packit 679830
        {
Packit 679830
            dict += dict_len - M4_MAX_OFFSET;
Packit 679830
            dict_len = M4_MAX_OFFSET;
Packit 679830
        }
Packit 679830
        dict_end = dict + dict_len;
Packit 679830
    }
Packit 679830
    else
Packit 679830
    {
Packit 679830
        dict_len = 0;
Packit 679830
        dict_end = NULL;
Packit 679830
    }
Packit 679830
#endif /* COPY_DICT */
Packit 679830
Packit 679830
    *out_len = 0;
Packit 679830
Packit 679830
    op = out;
Packit 679830
    ip = in;
Packit 679830
Packit 679830
    NEED_IP(1);
Packit 679830
    if (*ip > 17)
Packit 679830
    {
Packit 679830
        t = *ip++ - 17;
Packit 679830
        if (t < 4)
Packit 679830
            goto match_next;
Packit 679830
        assert(t > 0); NEED_OP(t); NEED_IP(t+3);
Packit 679830
        do *op++ = *ip++; while (--t > 0);
Packit 679830
        goto first_literal_run;
Packit 679830
    }
Packit 679830
Packit 679830
    for (;;)
Packit 679830
    {
Packit 679830
        NEED_IP(3);
Packit 679830
        t = *ip++;
Packit 679830
        if (t >= 16)
Packit 679830
            goto match;
Packit 679830
        /* a literal run */
Packit 679830
        if (t == 0)
Packit 679830
        {
Packit 679830
            while (*ip == 0)
Packit 679830
            {
Packit 679830
                t += 255;
Packit 679830
                ip++;
Packit 679830
                TEST_IV(t);
Packit 679830
                NEED_IP(1);
Packit 679830
            }
Packit 679830
            t += 15 + *ip++;
Packit 679830
        }
Packit 679830
        /* copy literals */
Packit 679830
        assert(t > 0); NEED_OP(t+3); NEED_IP(t+6);
Packit 679830
#if (LZO_OPT_UNALIGNED64) && (LZO_OPT_UNALIGNED32)
Packit 679830
        t += 3;
Packit 679830
        if (t >= 8) do
Packit 679830
        {
Packit 679830
            UA_COPY8(op,ip);
Packit 679830
            op += 8; ip += 8; t -= 8;
Packit 679830
        } while (t >= 8);
Packit 679830
        if (t >= 4)
Packit 679830
        {
Packit 679830
            UA_COPY4(op,ip);
Packit 679830
            op += 4; ip += 4; t -= 4;
Packit 679830
        }
Packit 679830
        if (t > 0)
Packit 679830
        {
Packit 679830
            *op++ = *ip++;
Packit 679830
            if (t > 1) { *op++ = *ip++; if (t > 2) { *op++ = *ip++; } }
Packit 679830
        }
Packit 679830
#elif (LZO_OPT_UNALIGNED32) || (LZO_ALIGNED_OK_4)
Packit 679830
#if !(LZO_OPT_UNALIGNED32)
Packit 679830
        if (PTR_ALIGNED2_4(op,ip))
Packit 679830
        {
Packit 679830
#endif
Packit 679830
        UA_COPY4(op,ip);
Packit 679830
        op += 4; ip += 4;
Packit 679830
        if (--t > 0)
Packit 679830
        {
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
                do *op++ = *ip++; while (--t > 0);
Packit 679830
        }
Packit 679830
#if !(LZO_OPT_UNALIGNED32)
Packit 679830
        }
Packit 679830
        else
Packit 679830
#endif
Packit 679830
#endif
Packit 679830
#if !(LZO_OPT_UNALIGNED32)
Packit 679830
        {
Packit 679830
            *op++ = *ip++; *op++ = *ip++; *op++ = *ip++;
Packit 679830
            do *op++ = *ip++; while (--t > 0);
Packit 679830
        }
Packit 679830
#endif
Packit 679830
Packit 679830
Packit 679830
first_literal_run:
Packit 679830
Packit 679830
Packit 679830
        t = *ip++;
Packit 679830
        if (t >= 16)
Packit 679830
            goto match;
Packit 679830
#if defined(COPY_DICT)
Packit 679830
#if defined(LZO1Z)
Packit 679830
        m_off = (1 + M2_MAX_OFFSET) + (t << 6) + (*ip++ >> 2);
Packit 679830
        last_m_off = m_off;
Packit 679830
#else
Packit 679830
        m_off = (1 + M2_MAX_OFFSET) + (t >> 2) + (*ip++ << 2);
Packit 679830
#endif
Packit 679830
        NEED_OP(3);
Packit 679830
        t = 3; COPY_DICT(t,m_off)
Packit 679830
#else /* !COPY_DICT */
Packit 679830
#if defined(LZO1Z)
Packit 679830
        t = (1 + M2_MAX_OFFSET) + (t << 6) + (*ip++ >> 2);
Packit 679830
        m_pos = op - t;
Packit 679830
        last_m_off = t;
Packit 679830
#else
Packit 679830
        m_pos = op - (1 + M2_MAX_OFFSET);
Packit 679830
        m_pos -= t >> 2;
Packit 679830
        m_pos -= *ip++ << 2;
Packit 679830
#endif
Packit 679830
        TEST_LB(m_pos); NEED_OP(3);
Packit 679830
        *op++ = *m_pos++; *op++ = *m_pos++; *op++ = *m_pos;
Packit 679830
#endif /* COPY_DICT */
Packit 679830
        goto match_done;
Packit 679830
Packit 679830
Packit 679830
        /* handle matches */
Packit 679830
        for (;;) {
Packit 679830
match:
Packit 679830
            if (t >= 64)                /* a M2 match */
Packit 679830
            {
Packit 679830
#if defined(COPY_DICT)
Packit 679830
#if defined(LZO1X)
Packit 679830
                m_off = 1 + ((t >> 2) & 7) + (*ip++ << 3);
Packit 679830
                t = (t >> 5) - 1;
Packit 679830
#elif defined(LZO1Y)
Packit 679830
                m_off = 1 + ((t >> 2) & 3) + (*ip++ << 2);
Packit 679830
                t = (t >> 4) - 3;
Packit 679830
#elif defined(LZO1Z)
Packit 679830
                m_off = t & 0x1f;
Packit 679830
                if (m_off >= 0x1c)
Packit 679830
                    m_off = last_m_off;
Packit 679830
                else
Packit 679830
                {
Packit 679830
                    m_off = 1 + (m_off << 6) + (*ip++ >> 2);
Packit 679830
                    last_m_off = m_off;
Packit 679830
                }
Packit 679830
                t = (t >> 5) - 1;
Packit 679830
#endif
Packit 679830
#else /* !COPY_DICT */
Packit 679830
#if defined(LZO1X)
Packit 679830
                m_pos = op - 1;
Packit 679830
                m_pos -= (t >> 2) & 7;
Packit 679830
                m_pos -= *ip++ << 3;
Packit 679830
                t = (t >> 5) - 1;
Packit 679830
#elif defined(LZO1Y)
Packit 679830
                m_pos = op - 1;
Packit 679830
                m_pos -= (t >> 2) & 3;
Packit 679830
                m_pos -= *ip++ << 2;
Packit 679830
                t = (t >> 4) - 3;
Packit 679830
#elif defined(LZO1Z)
Packit 679830
                {
Packit 679830
                    lzo_uint off = t & 0x1f;
Packit 679830
                    m_pos = op;
Packit 679830
                    if (off >= 0x1c)
Packit 679830
                    {
Packit 679830
                        assert(last_m_off > 0);
Packit 679830
                        m_pos -= last_m_off;
Packit 679830
                    }
Packit 679830
                    else
Packit 679830
                    {
Packit 679830
                        off = 1 + (off << 6) + (*ip++ >> 2);
Packit 679830
                        m_pos -= off;
Packit 679830
                        last_m_off = off;
Packit 679830
                    }
Packit 679830
                }
Packit 679830
                t = (t >> 5) - 1;
Packit 679830
#endif
Packit 679830
                TEST_LB(m_pos); assert(t > 0); NEED_OP(t+3-1);
Packit 679830
                goto copy_match;
Packit 679830
#endif /* COPY_DICT */
Packit 679830
            }
Packit 679830
            else if (t >= 32)           /* a M3 match */
Packit 679830
            {
Packit 679830
                t &= 31;
Packit 679830
                if (t == 0)
Packit 679830
                {
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 += 31 + *ip++;
Packit 679830
                    NEED_IP(2);
Packit 679830
                }
Packit 679830
#if defined(COPY_DICT)
Packit 679830
#if defined(LZO1Z)
Packit 679830
                m_off = 1 + (ip[0] << 6) + (ip[1] >> 2);
Packit 679830
                last_m_off = m_off;
Packit 679830
#else
Packit 679830
                m_off = 1 + (ip[0] >> 2) + (ip[1] << 6);
Packit 679830
#endif
Packit 679830
#else /* !COPY_DICT */
Packit 679830
#if defined(LZO1Z)
Packit 679830
                {
Packit 679830
                    lzo_uint off = 1 + (ip[0] << 6) + (ip[1] >> 2);
Packit 679830
                    m_pos = op - off;
Packit 679830
                    last_m_off = off;
Packit 679830
                }
Packit 679830
#elif (LZO_OPT_UNALIGNED16) && (LZO_ABI_LITTLE_ENDIAN)
Packit 679830
                m_pos = op - 1;
Packit 679830
                m_pos -= UA_GET_LE16(ip) >> 2;
Packit 679830
#else
Packit 679830
                m_pos = op - 1;
Packit 679830
                m_pos -= (ip[0] >> 2) + (ip[1] << 6);
Packit 679830
#endif
Packit 679830
#endif /* COPY_DICT */
Packit 679830
                ip += 2;
Packit 679830
            }
Packit 679830
            else if (t >= 16)           /* a M4 match */
Packit 679830
            {
Packit 679830
#if defined(COPY_DICT)
Packit 679830
                m_off = (t & 8) << 11;
Packit 679830
#else /* !COPY_DICT */
Packit 679830
                m_pos = op;
Packit 679830
                m_pos -= (t & 8) << 11;
Packit 679830
#endif /* COPY_DICT */
Packit 679830
                t &= 7;
Packit 679830
                if (t == 0)
Packit 679830
                {
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 += 7 + *ip++;
Packit 679830
                    NEED_IP(2);
Packit 679830
                }
Packit 679830
#if defined(COPY_DICT)
Packit 679830
#if defined(LZO1Z)
Packit 679830
                m_off += (ip[0] << 6) + (ip[1] >> 2);
Packit 679830
#else
Packit 679830
                m_off += (ip[0] >> 2) + (ip[1] << 6);
Packit 679830
#endif
Packit 679830
                ip += 2;
Packit 679830
                if (m_off == 0)
Packit 679830
                    goto eof_found;
Packit 679830
                m_off += 0x4000;
Packit 679830
#if defined(LZO1Z)
Packit 679830
                last_m_off = m_off;
Packit 679830
#endif
Packit 679830
#else /* !COPY_DICT */
Packit 679830
#if defined(LZO1Z)
Packit 679830
                m_pos -= (ip[0] << 6) + (ip[1] >> 2);
Packit 679830
#elif (LZO_OPT_UNALIGNED16) && (LZO_ABI_LITTLE_ENDIAN)
Packit 679830
                m_pos -= UA_GET_LE16(ip) >> 2;
Packit 679830
#else
Packit 679830
                m_pos -= (ip[0] >> 2) + (ip[1] << 6);
Packit 679830
#endif
Packit 679830
                ip += 2;
Packit 679830
                if (m_pos == op)
Packit 679830
                    goto eof_found;
Packit 679830
                m_pos -= 0x4000;
Packit 679830
#if defined(LZO1Z)
Packit 679830
                last_m_off = pd((const lzo_bytep)op, m_pos);
Packit 679830
#endif
Packit 679830
#endif /* COPY_DICT */
Packit 679830
            }
Packit 679830
            else                            /* a M1 match */
Packit 679830
            {
Packit 679830
#if defined(COPY_DICT)
Packit 679830
#if defined(LZO1Z)
Packit 679830
                m_off = 1 + (t << 6) + (*ip++ >> 2);
Packit 679830
                last_m_off = m_off;
Packit 679830
#else
Packit 679830
                m_off = 1 + (t >> 2) + (*ip++ << 2);
Packit 679830
#endif
Packit 679830
                NEED_OP(2);
Packit 679830
                t = 2; COPY_DICT(t,m_off)
Packit 679830
#else /* !COPY_DICT */
Packit 679830
#if defined(LZO1Z)
Packit 679830
                t = 1 + (t << 6) + (*ip++ >> 2);
Packit 679830
                m_pos = op - t;
Packit 679830
                last_m_off = t;
Packit 679830
#else
Packit 679830
                m_pos = op - 1;
Packit 679830
                m_pos -= t >> 2;
Packit 679830
                m_pos -= *ip++ << 2;
Packit 679830
#endif
Packit 679830
                TEST_LB(m_pos); NEED_OP(2);
Packit 679830
                *op++ = *m_pos++; *op++ = *m_pos;
Packit 679830
#endif /* COPY_DICT */
Packit 679830
                goto match_done;
Packit 679830
            }
Packit 679830
Packit 679830
            /* copy match */
Packit 679830
#if defined(COPY_DICT)
Packit 679830
Packit 679830
            NEED_OP(t+3-1);
Packit 679830
            t += 3-1; COPY_DICT(t,m_off)
Packit 679830
Packit 679830
#else /* !COPY_DICT */
Packit 679830
Packit 679830
            TEST_LB(m_pos); assert(t > 0); NEED_OP(t+3-1);
Packit 679830
#if (LZO_OPT_UNALIGNED64) && (LZO_OPT_UNALIGNED32)
Packit 679830
            if (op - m_pos >= 8)
Packit 679830
            {
Packit 679830
                t += (3 - 1);
Packit 679830
                if (t >= 8) do
Packit 679830
                {
Packit 679830
                    UA_COPY8(op,m_pos);
Packit 679830
                    op += 8; m_pos += 8; t -= 8;
Packit 679830
                } while (t >= 8);
Packit 679830
                if (t >= 4)
Packit 679830
                {
Packit 679830
                    UA_COPY4(op,m_pos);
Packit 679830
                    op += 4; m_pos += 4; t -= 4;
Packit 679830
                }
Packit 679830
                if (t > 0)
Packit 679830
                {
Packit 679830
                    *op++ = m_pos[0];
Packit 679830
                    if (t > 1) { *op++ = m_pos[1]; if (t > 2) { *op++ = m_pos[2]; } }
Packit 679830
                }
Packit 679830
            }
Packit 679830
            else
Packit 679830
#elif (LZO_OPT_UNALIGNED32) || (LZO_ALIGNED_OK_4)
Packit 679830
#if !(LZO_OPT_UNALIGNED32)
Packit 679830
            if (t >= 2 * 4 - (3 - 1) && PTR_ALIGNED2_4(op,m_pos))
Packit 679830
            {
Packit 679830
                assert((op - m_pos) >= 4);  /* both pointers are aligned */
Packit 679830
#else
Packit 679830
            if (t >= 2 * 4 - (3 - 1) && (op - m_pos) >= 4)
Packit 679830
            {
Packit 679830
#endif
Packit 679830
                UA_COPY4(op,m_pos);
Packit 679830
                op += 4; m_pos += 4; t -= 4 - (3 - 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_match:
Packit 679830
                *op++ = *m_pos++; *op++ = *m_pos++;
Packit 679830
                do *op++ = *m_pos++; while (--t > 0);
Packit 679830
            }
Packit 679830
Packit 679830
#endif /* COPY_DICT */
Packit 679830
Packit 679830
match_done:
Packit 679830
#if defined(LZO1Z)
Packit 679830
            t = ip[-1] & 3;
Packit 679830
#else
Packit 679830
            t = ip[-2] & 3;
Packit 679830
#endif
Packit 679830
            if (t == 0)
Packit 679830
                break;
Packit 679830
Packit 679830
            /* copy literals */
Packit 679830
match_next:
Packit 679830
            assert(t > 0); assert(t < 4); NEED_OP(t); NEED_IP(t+3);
Packit 679830
#if 0
Packit 679830
            do *op++ = *ip++; while (--t > 0);
Packit 679830
#else
Packit 679830
            *op++ = *ip++;
Packit 679830
            if (t > 1) { *op++ = *ip++; if (t > 2) { *op++ = *ip++; } }
Packit 679830
#endif
Packit 679830
            t = *ip++;
Packit 679830
        }
Packit 679830
    }
Packit 679830
Packit 679830
eof_found:
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