Algorithm 版 (精华区)

发信人: lofe ()感激生活(), 信区: Algorithm
标  题: ar002--huf.c
发信站: 哈工大紫丁香 (Sun Sep  3 08:27:50 2000), 转信

/***********************************************************
        huf.c -- static Huffman
***********************************************************/
#include <stdlib.h>
#include "ar.h"

#define NP (DICBIT + 1)
#define NT (CODE_BIT + 3)
#define PBIT 4  /* smallest integer such that (1U << PBIT) > NP */
#define TBIT 5  /* smallest integer such that (1U << TBIT) > NT */
#if NT > NP
        #define NPT NT
#else
        #define NPT NP
#endif

ushort left[2 * NC - 1], right[2 * NC - 1];
static uchar *buf, c_len[NC], pt_len[NPT];
static uint   bufsiz = 0, blocksize;
static ushort c_freq[2 * NC - 1], c_table[4096], c_code[NC],
                          p_freq[2 * NP - 1], pt_table[256], pt_code[NPT],
                          t_freq[2 * NT - 1];

/***** encoding *****/

static void count_t_freq(void)
{
        int i, k, n, count;

        for (i = 0; i < NT; i++) t_freq[i] = 0;
        n = NC;
        while (n > 0 && c_len[n - 1] == 0) n--;
        i = 0;
        while (i < n) {
                k = c_len[i++];
                if (k == 0) {
                        count = 1;
                        while (i < n && c_len[i] == 0) {  i++;  count++;  }
                        if (count <= 2) t_freq[0] += count;
                        else if (count <= 18) t_freq[1]++;
                        else if (count == 19) {  t_freq[0]++;  t_freq[1]++;  }
                        else t_freq[2]++;
                } else t_freq[k + 2]++;
        }
}

static void write_pt_len(int n, int nbit, int i_special)
{
        int i, k;

        while (n > 0 && pt_len[n - 1] == 0) n--;
        putbits(nbit, n);
        i = 0;
        while (i < n) {
                k = pt_len[i++];
                if (k <= 6) putbits(3, k);
                else putbits(k - 3, (1U << (k - 3)) - 2);
                if (i == i_special) {
                        while (i < 6 && pt_len[i] == 0) i++;
                        putbits(2, (i - 3) & 3);
                }
        }
}

static void write_c_len(void)
{
        int i, k, n, count;

        n = NC;
        while (n > 0 && c_len[n - 1] == 0) n--;
        putbits(CBIT, n);
        i = 0;
        while (i < n) {
                k = c_len[i++];
                if (k == 0) {
                        count = 1;
                        while (i < n && c_len[i] == 0) {  i++;  count++;  }
                        if (count <= 2) {
                                for (k = 0; k < count; k++)
                                        putbits(pt_len[0], pt_code[0]);
                        } else if (count <= 18) {
                                putbits(pt_len[1], pt_code[1]);
                                putbits(4, count - 3);
                        } else if (count == 19) {
                                putbits(pt_len[0], pt_code[0]);
                                putbits(pt_len[1], pt_code[1]);
                                putbits(4, 15);
                        } else {
                                putbits(pt_len[2], pt_code[2]);
                                putbits(CBIT, count - 20);
                        }
                } else putbits(pt_len[k + 2], pt_code[k + 2]);
        }
}

static void encode_c(int c)
{
        putbits(c_len[c], c_code[c]);
}

static void encode_p(uint p)
{
        uint c, q;

        c = 0;  q = p;  while (q) {  q >>= 1;  c++;  }
        putbits(pt_len[c], pt_code[c]);
        if (c > 1) putbits(c - 1, p & (0xFFFFU >> (17 - c)));
}

static void send_block(void)
{
        uint i, k, flags, root, pos, size;

        root = make_tree(NC, c_freq, c_len, c_code);
        size = c_freq[root];  putbits(16, size);
        if (root >= NC) {
                count_t_freq();
                root = make_tree(NT, t_freq, pt_len, pt_code);
                if (root >= NT) {
                        write_pt_len(NT, TBIT, 3);
                } else {
                        putbits(TBIT, 0);  putbits(TBIT, root);
                }
                write_c_len();
        } else {
        putbits(TBIT, 0);  putbits(TBIT, 0);
                putbits(CBIT, 0);  putbits(CBIT, root);
        }
        root = make_tree(NP, p_freq, pt_len, pt_code);
        if (root >= NP) {
                write_pt_len(NP, PBIT, -1);
        } else {
                putbits(PBIT, 0);  putbits(PBIT, root);
        }
        pos = 0;
        for (i = 0; i < size; i++) {
                if (i % CHAR_BIT == 0) flags = buf[pos++];  else flags <<= 1;
                if (flags & (1U << (CHAR_BIT - 1))) {
                        encode_c(buf[pos++] + (1U << CHAR_BIT));
                        k = buf[pos++] << CHAR_BIT;  k += buf[pos++];
                        encode_p(k);
                } else encode_c(buf[pos++]);
                if (unpackable) return;
        }
        for (i = 0; i < NC; i++) c_freq[i] = 0;
        for (i = 0; i < NP; i++) p_freq[i] = 0;
}

static uint output_pos, output_mask;

void output(uint c, uint p)
{
        static uint cpos;

        if ((output_mask >>= 1) == 0) {
                output_mask = 1U << (CHAR_BIT - 1);
                if (output_pos >= bufsiz - 3 * CHAR_BIT) {
                        send_block();
                        if (unpackable) return;
                        output_pos = 0;
                }
                cpos = output_pos++;  buf[cpos] = 0;
        }
        buf[output_pos++] = (uchar) c;  c_freq[c]++;
        if (c >= (1U << CHAR_BIT)) {
                buf[cpos] |= output_mask;
                buf[output_pos++] = (uchar)(p >> CHAR_BIT);
                buf[output_pos++] = (uchar) p;
                c = 0;  while (p) {  p >>= 1;  c++;  }
                p_freq[c]++;
        }
}

void huf_encode_start(void)
{
        int i;

        if (bufsiz == 0) {
        bufsiz = 16 * 1024U;
                while ((buf = malloc(bufsiz)) == NULL) {
                        bufsiz = (bufsiz / 10U) * 9U;
                        if (bufsiz < 4 * 1024U) error("Out of memory.");
                }
        }
        buf[0] = 0;
        for (i = 0; i < NC; i++) c_freq[i] = 0;
        for (i = 0; i < NP; i++) p_freq[i] = 0;
        output_pos = output_mask = 0;
        init_putbits();
}

void huf_encode_end(void)
{
        if (! unpackable) {
                send_block();
                putbits(CHAR_BIT - 1, 0);  /* flush remaining bits */
        }
}

/***** decoding *****/

static void read_pt_len(int nn, int nbit, int i_special)
{
        int i, c, n;
        uint mask;

        n = getbits(nbit);
        if (n == 0) {
                c = getbits(nbit);
                for (i = 0; i < nn; i++) pt_len[i] = 0;
                for (i = 0; i < 256; i++) pt_table[i] = c;
        } else {
                i = 0;
                while (i < n) {
                        c = bitbuf >> (BITBUFSIZ - 3);
                        if (c == 7) {
                                mask = 1U << (BITBUFSIZ - 1 - 3);
                                while (mask & bitbuf) {  mask >>= 1;  c++;  }
                        }
                        fillbuf((c < 7) ? 3 : c - 3);
                        pt_len[i++] = c;
                        if (i == i_special) {
                                c = getbits(2);
                                while (--c >= 0) pt_len[i++] = 0;
                        }
                }
                while (i < nn) pt_len[i++] = 0;
                make_table(nn, pt_len, 8, pt_table);
        }
}

static void read_c_len(void)
{
        int i, c, n;
        uint mask;

        n = getbits(CBIT);
        if (n == 0) {
                c = getbits(CBIT);
                for (i = 0; i < NC; i++) c_len[i] = 0;
                for (i = 0; i < 4096; i++) c_table[i] = c;
        } else {
                i = 0;
                while (i < n) {
                        c = pt_table[bitbuf >> (BITBUFSIZ - 8)];
                        if (c >= NT) {
                                mask = 1U << (BITBUFSIZ - 1 - 8);
                                do {
                                        if (bitbuf & mask) c = right[c];
                                        else               c = left [c];
                                        mask >>= 1;
                                } while (c >= NT);
                        }
                        fillbuf(pt_len[c]);
                        if (c <= 2) {
                                if      (c == 0) c = 1;
                                else if (c == 1) c = getbits(4) + 3;
                                else             c = getbits(CBIT) + 20;
                                while (--c >= 0) c_len[i++] = 0;
                        } else c_len[i++] = c - 2;
                }
                while (i < NC) c_len[i++] = 0;
                make_table(NC, c_len, 12, c_table);
        }
}

uint decode_c(void)
{
        uint j, mask;

        if (blocksize == 0) {
                blocksize = getbits(16);
                read_pt_len(NT, TBIT, 3);
                read_c_len();
                read_pt_len(NP, PBIT, -1);
        }
        blocksize--;
        j = c_table[bitbuf >> (BITBUFSIZ - 12)];
        if (j >= NC) {
                mask = 1U << (BITBUFSIZ - 1 - 12);
                do {
                        if (bitbuf & mask) j = right[j];
                        else               j = left [j];
                        mask >>= 1;
                } while (j >= NC);
        }
        fillbuf(c_len[j]);
        return j;
}

uint decode_p(void)
{
        uint j, mask;

        j = pt_table[bitbuf >> (BITBUFSIZ - 8)];
        if (j >= NP) {
                mask = 1U << (BITBUFSIZ - 1 - 8);
                do {
                        if (bitbuf & mask) j = right[j];
                        else               j = left [j];
                        mask >>= 1;
                } while (j >= NP);
        }
        fillbuf(pt_len[j]);
        if (j != 0) j = (1U << (j - 1)) + getbits(j - 1);
        return j;
}

void huf_decode_start(void)
{
        init_getbits();  blocksize = 0;
}

--
We Are the World!
                        infosite@263.net
※ 修改:.haojs 于 Sep  3 08:25:21 修改本文.[FROM: bbs.hit.edu.cn]
--
※ 转寄:.武汉白云黄鹤站 bbs.whnet.edu.cn.[FROM: bbs.hit.edu.cn]

--
☆ 来源:.哈工大紫丁香 bbs.hit.edu.cn.[FROM: haojs.bbs@bbs.whnet.]
[百宝箱] [返回首页] [上级目录] [根目录] [返回顶部] [刷新] [返回]
Powered by KBS BBS 2.0 (http://dev.kcn.cn)
页面执行时间:207.241毫秒