Blame src/addressbook/mspack/readhuff.h

Packit Service 6d57b4
/* This file is part of libmspack.
Packit Service 6d57b4
 * (C) 2003-2010 Stuart Caie.
Packit Service 6d57b4
 *
Packit Service 6d57b4
 * libmspack is free software; you can redistribute it and/or modify it under
Packit Service 6d57b4
 * the terms of the GNU Lesser General Public License (LGPL) version 2.1
Packit Service 6d57b4
 *
Packit Service 6d57b4
 * For further details, see the file COPYING.LIB distributed with libmspack
Packit Service 6d57b4
 */
Packit Service 6d57b4
Packit Service 6d57b4
#ifndef MSPACK_READHUFF_H
Packit Service 6d57b4
#define MSPACK_READHUFF_H 1
Packit Service 6d57b4
Packit Service 6d57b4
/* This implements a fast Huffman tree decoding system.
Packit Service 6d57b4
 */
Packit Service 6d57b4
Packit Service 6d57b4
#if !(defined(BITS_ORDER_MSB) || defined(BITS_ORDER_LSB))
Packit Service 6d57b4
# error "readhuff.h is used in conjunction with readbits.h, include that first"
Packit Service 6d57b4
#endif
Packit Service 6d57b4
#if !(defined(TABLEBITS) && defined(MAXSYMBOLS))
Packit Service 6d57b4
# error "define TABLEBITS(tbl) and MAXSYMBOLS(tbl) before using readhuff.h"
Packit Service 6d57b4
#endif
Packit Service 6d57b4
#if !(defined(HUFF_TABLE) && defined(HUFF_LEN))
Packit Service 6d57b4
# error "define HUFF_TABLE(tbl) and HUFF_LEN(tbl) before using readhuff.h"
Packit Service 6d57b4
#endif
Packit Service 6d57b4
#ifndef HUFF_ERROR
Packit Service 6d57b4
# error "define HUFF_ERROR before using readhuff.h"
Packit Service 6d57b4
#endif
Packit Service 6d57b4
#ifndef HUFF_MAXBITS
Packit Service 6d57b4
# define HUFF_MAXBITS 16
Packit Service 6d57b4
#endif
Packit Service 6d57b4
Packit Service 6d57b4
/* Decodes the next huffman symbol from the input bitstream into var.
Packit Service 6d57b4
 * Do not use this macro on a table unless build_decode_table() succeeded.
Packit Service 6d57b4
 */
Packit Service 6d57b4
#define READ_HUFFSYM(tbl, var) do {			\
Packit Service 6d57b4
    ENSURE_BITS(HUFF_MAXBITS);				\
Packit Service 6d57b4
    sym = HUFF_TABLE(tbl, PEEK_BITS(TABLEBITS(tbl)));	\
Packit Service 6d57b4
    if (sym >= MAXSYMBOLS(tbl))	HUFF_TRAVERSE(tbl);	\
Packit Service 6d57b4
    (var) = sym;					\
Packit Service 6d57b4
    i = HUFF_LEN(tbl, sym);				\
Packit Service 6d57b4
    REMOVE_BITS(i);					\
Packit Service 6d57b4
} while (0)
Packit Service 6d57b4
Packit Service 6d57b4
#ifdef BITS_ORDER_LSB
Packit Service 6d57b4
# define HUFF_TRAVERSE(tbl) do {			\
Packit Service 6d57b4
    i = TABLEBITS(tbl) - 1;				\
Packit Service 6d57b4
    do {						\
Packit Service 6d57b4
	if (i++ > HUFF_MAXBITS) HUFF_ERROR;		\
Packit Service 6d57b4
	sym = HUFF_TABLE(tbl,				\
Packit Service 6d57b4
            (sym << 1) | ((bit_buffer >> i) & 1));	\
Packit Service 6d57b4
    } while (sym >= MAXSYMBOLS(tbl));			\
Packit Service 6d57b4
} while (0)
Packit Service 6d57b4
#else
Packit Service 6d57b4
#define HUFF_TRAVERSE(tbl) do {				\
Packit Service 6d57b4
    i = 1 << (BITBUF_WIDTH - TABLEBITS(tbl));		\
Packit Service 6d57b4
    do {						\
Packit Service 6d57b4
	if ((i >>= 1) == 0) HUFF_ERROR;			\
Packit Service 6d57b4
	sym = HUFF_TABLE(tbl,				\
Packit Service 6d57b4
	    (sym << 1) | ((bit_buffer & i) ? 1 : 0));	\
Packit Service 6d57b4
    } while (sym >= MAXSYMBOLS(tbl));			\
Packit Service 6d57b4
} while (0)
Packit Service 6d57b4
#endif
Packit Service 6d57b4
Packit Service 6d57b4
/* make_decode_table(nsyms, nbits, length[], table[])
Packit Service 6d57b4
 *
Packit Service 6d57b4
 * This function was originally coded by David Tritscher.
Packit Service 6d57b4
 * It builds a fast huffman decoding table from
Packit Service 6d57b4
 * a canonical huffman code lengths table.
Packit Service 6d57b4
 *
Packit Service 6d57b4
 * nsyms  = total number of symbols in this huffman tree.
Packit Service 6d57b4
 * nbits  = any symbols with a code length of nbits or less can be decoded
Packit Service 6d57b4
 *          in one lookup of the table.
Packit Service 6d57b4
 * length = A table to get code lengths from [0 to nsyms-1]
Packit Service 6d57b4
 * table  = The table to fill up with decoded symbols and pointers.
Packit Service 6d57b4
 *          Should be ((1<
Packit Service 6d57b4
 *
Packit Service 6d57b4
 * Returns 0 for OK or 1 for error
Packit Service 6d57b4
 */
Packit Service 6d57b4
static int make_decode_table(unsigned int nsyms, unsigned int nbits,
Packit Service 6d57b4
			     unsigned char *length, unsigned short *table)
Packit Service 6d57b4
{
Packit Service 6d57b4
    register unsigned short sym, next_symbol;
Packit Service 6d57b4
    register unsigned int leaf, fill;
Packit Service 6d57b4
#ifdef BITS_ORDER_LSB
Packit Service 6d57b4
    register unsigned int reverse;
Packit Service 6d57b4
#endif
Packit Service 6d57b4
    register unsigned char bit_num;
Packit Service 6d57b4
    unsigned int pos         = 0; /* the current position in the decode table */
Packit Service 6d57b4
    unsigned int table_mask  = 1 << nbits;
Packit Service 6d57b4
    unsigned int bit_mask    = table_mask >> 1; /* don't do 0 length codes */
Packit Service 6d57b4
Packit Service 6d57b4
    /* fill entries for codes short enough for a direct mapping */
Packit Service 6d57b4
    for (bit_num = 1; bit_num <= nbits; bit_num++) {
Packit Service 6d57b4
	for (sym = 0; sym < nsyms; sym++) {
Packit Service 6d57b4
	    if (length[sym] != bit_num) continue;
Packit Service 6d57b4
#ifdef BITS_ORDER_MSB
Packit Service 6d57b4
	    leaf = pos;
Packit Service 6d57b4
#else
Packit Service 6d57b4
	    /* reverse the significant bits */
Packit Service 6d57b4
	    fill = length[sym]; reverse = pos >> (nbits - fill); leaf = 0;
Packit Service 6d57b4
	    do {leaf <<= 1; leaf |= reverse & 1; reverse >>= 1;} while (--fill);
Packit Service 6d57b4
#endif
Packit Service 6d57b4
Packit Service 6d57b4
	    if((pos += bit_mask) > table_mask) return 1; /* table overrun */
Packit Service 6d57b4
Packit Service 6d57b4
	    /* fill all possible lookups of this symbol with the symbol itself */
Packit Service 6d57b4
#ifdef BITS_ORDER_MSB
Packit Service 6d57b4
	    for (fill = bit_mask; fill-- > 0;) table[leaf++] = sym;
Packit Service 6d57b4
#else
Packit Service 6d57b4
	    fill = bit_mask; next_symbol = 1 << bit_num;
Packit Service 6d57b4
	    do { table[leaf] = sym; leaf += next_symbol; } while (--fill);
Packit Service 6d57b4
#endif
Packit Service 6d57b4
	}
Packit Service 6d57b4
	bit_mask >>= 1;
Packit Service 6d57b4
    }
Packit Service 6d57b4
Packit Service 6d57b4
    /* exit with success if table is now complete */
Packit Service 6d57b4
    if (pos == table_mask) return 0;
Packit Service 6d57b4
Packit Service 6d57b4
    /* mark all remaining table entries as unused */
Packit Service 6d57b4
    for (sym = pos; sym < table_mask; sym++) {
Packit Service 6d57b4
#ifdef BITS_ORDER_MSB
Packit Service 6d57b4
	table[sym] = 0xFFFF;
Packit Service 6d57b4
#else
Packit Service 6d57b4
	reverse = sym; leaf = 0; fill = nbits;
Packit Service 6d57b4
	do { leaf <<= 1; leaf |= reverse & 1; reverse >>= 1; } while (--fill);
Packit Service 6d57b4
	table[leaf] = 0xFFFF;
Packit Service 6d57b4
#endif
Packit Service 6d57b4
    }
Packit Service 6d57b4
Packit Service 6d57b4
    /* next_symbol = base of allocation for long codes */
Packit Service 6d57b4
    next_symbol = ((table_mask >> 1) < nsyms) ? nsyms : (table_mask >> 1);
Packit Service 6d57b4
Packit Service 6d57b4
    /* give ourselves room for codes to grow by up to 16 more bits.
Packit Service 6d57b4
     * codes now start at bit nbits+16 and end at (nbits+16-codelength) */
Packit Service 6d57b4
    pos <<= 16;
Packit Service 6d57b4
    table_mask <<= 16;
Packit Service 6d57b4
    bit_mask = 1 << 15;
Packit Service 6d57b4
Packit Service 6d57b4
    for (bit_num = nbits+1; bit_num <= HUFF_MAXBITS; bit_num++) {
Packit Service 6d57b4
	for (sym = 0; sym < nsyms; sym++) {
Packit Service 6d57b4
	    if (length[sym] != bit_num) continue;
Packit Service 6d57b4
Packit Service 6d57b4
#ifdef BITS_ORDER_MSB
Packit Service 6d57b4
	    leaf = pos >> 16;
Packit Service 6d57b4
#else
Packit Service 6d57b4
	    /* leaf = the first nbits of the code, reversed */
Packit Service 6d57b4
	    reverse = pos >> 16; leaf = 0; fill = nbits;
Packit Service 6d57b4
	    do {leaf <<= 1; leaf |= reverse & 1; reverse >>= 1;} while (--fill);
Packit Service 6d57b4
#endif
Packit Service 6d57b4
	    for (fill = 0; fill < (bit_num - nbits); fill++) {
Packit Service 6d57b4
		/* if this path hasn't been taken yet, 'allocate' two entries */
Packit Service 6d57b4
		if (table[leaf] == 0xFFFF) {
Packit Service 6d57b4
		    table[(next_symbol << 1)     ] = 0xFFFF;
Packit Service 6d57b4
		    table[(next_symbol << 1) + 1 ] = 0xFFFF;
Packit Service 6d57b4
		    table[leaf] = next_symbol++;
Packit Service 6d57b4
		}
Packit Service 6d57b4
Packit Service 6d57b4
		/* follow the path and select either left or right for next bit */
Packit Service 6d57b4
		leaf = table[leaf] << 1;
Packit Service 6d57b4
		if ((pos >> (15-fill)) & 1) leaf++;
Packit Service 6d57b4
	    }
Packit Service 6d57b4
	    table[leaf] = sym;
Packit Service 6d57b4
Packit Service 6d57b4
	    if ((pos += bit_mask) > table_mask) return 1; /* table overflow */
Packit Service 6d57b4
	}
Packit Service 6d57b4
	bit_mask >>= 1;
Packit Service 6d57b4
    }
Packit Service 6d57b4
Packit Service 6d57b4
    /* full table? */
Packit Service 6d57b4
    return (pos == table_mask) ? 0 : 1;
Packit Service 6d57b4
}
Packit Service 6d57b4
#endif