Algorithm 版 (精华区)

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

#include <stdio.h>
#include <mem.h>
#include <alloc.h>
#include <stdlib.h>

#include "comp.h"
#include "arith.h"


#define MAX_CUM_FREQ   0x4000   /* maximum cumulative frequency */

struct model

{
        int initial_level;
        int match_level;

        unsigned base_count;
        unsigned sym_freq;

        unsigned order_cum_freq [MAX_ORDER + 2];
        unsigned order_sym_count [MAX_ORDER + 2];
};



/*
        Define circular dictionary large enough to hold complete
        set of active input symbols.

        Extra entries are included in the dictionary corresponding
        to a string of length equal to the maximum order to facilitate
        searching across the end of the dictionary.  This extra space
        is allocated at the front of the dictionary, so that the first
        few entries are never referenced directly.

        An additional table is allocated consisting of one word entries
        used to link equivalent dictionary entries where a hash table is
        used to locate the start of each search chain.

        The last character of the test string is used locate the
        initial hash table entry.  The hash table entries are stored
        as an extension to the link table.

        The link table is accessed through the use of macros to allow
        easy access to this table when its size excceds 64K.
*/

#ifndef FAR_TABLES

static unsigned char dict [MAX_DICT];
static unsigned char base_level [MAX_DICT];
static unsigned int next_dict [MAX_DICT + HTBL1_SIZE];

#define DICT_WORD_PTR(i) ((unsigned *) (&dict [i]));
#define NEXT_DICT(i) next_dict [i]

#else

static unsigned char far *dict;
static unsigned char far *base_level;
#define DICT_WORD_PTR(i) ((unsigned far *) (&dict [i]));

#ifdef SPLIT_TABLES

static unsigned int far *next_dict [2];
#define NEXT_DICT(i) next_dict [i & 1] [i >> 1]

#else

static unsigned int far *next_dict;
#define NEXT_DICT(i) next_dict [i]

#endif
#endif


static unsigned int node;
static int new_symbol_count;
static unsigned int last_search_node;

static int max_order;
static int max_pos_size;

/*
        Define table of entries for each order giving number of symbols
        and total cumulative frequency for each order
*/

static struct model active_state;
static struct model last_state;


/*
        Define set of save areas for the state at the potential start any string
        The areas are used in rotation until a substring is matched equal to
        the minimum string length.  Normal symbol compression is performed
        during the time that the substring matches the input data.
                
        When the first non matching character is encountered, the lengths of
        the string and the compressed symbols are compared.  If the string
        length is less, the output is repositioned as specified by the
        state at the start of the string, and the string selection sequence
        overwrites the output.
*/

static struct

{
        int i;
        struct coder_state cs;
        struct model mod;
} save_state [MAX_STR_SAVE];


static int save_state_index;
static unsigned int string_pos;
static int string_len;
static int string_start;
static int skip_count;

/*
        States used while testing for text string replacements
*/

static enum
{
        STRING_WAIT,  STRING_SEARCH,  STRING_START,  STRING_ACTIVE,
        STRING_COMP,
} string_state;


/*
        Define combined set of tables giving the order and
        frequency count for each symbol encountered during dictionary scan
        Also accumulate base frequency values for each order
*/

static unsigned char level [MAX_SYM];
static unsigned int freq [MAX_SYM];
static unsigned int base_freq [MAX_ORDER + 2];

static unsigned int dup_count = 0;
static int dup_char = -1;

/*
        Build frequency table for zero and default orders
        Used to initialize state tables at start of each dictionary scan

        Always contains nonzero count for every symbol
        Includes order for each symbol selecting zero or default orders
        Initially set to all defaults with symbol frequency one
*/

static unsigned int sym_count_zero;
static unsigned int cum_freq_zero;
static unsigned int freq_zero [MAX_SYM];
static unsigned char order_zero [MAX_SYM];


/*
        Prototypes
*/

static void clear_context (void);
static void scan_dict (int);
static void scale_freq_tbl (int, int);
static void calc_symbol_freq (int);
static void select_output_symbol (void);
static int decode_symbol (void);
static void update_model (int);

static void encode_symbol (struct model *);
static void generate_switch_code (struct model *);
static void generate_symbol_code (struct model *);
static void generate_value (unsigned, unsigned);

static int start_decode_string (void);
static int decode_active_state (void);
static int decode_string_char (void);
static unsigned decode_value (unsigned);

static unsigned switch_char_freq (unsigned, unsigned);

static void delete_dict_entry (int);
static void test_string_state (void);
static void clear_text_string (void);

static void check_string_cont (void);
static void test_string_start (unsigned pos, int n);
static void test_string_resume (unsigned pos, int n);
static void replace_text_string (void);

static int log2 (unsigned);
static void scale_binary_value (long, int *, unsigned *);
static void update_bit_len (unsigned, unsigned, int *, unsigned *);
static int find_string_len (void);


/*
        Initialize model at start of compression or expansion
        Alocate and initialize tables
*/

void InitModel (int n)

{
        unsigned i;

        InitCoder ();

        active_state.initial_level = 1;
        node = MAX_ORDER;
        max_pos_size = log2 (NDICT - 1) + 1;
        max_order = n;
        if (max_order == 0) max_order = 1;

        sym_count_zero = 0;
        cum_freq_zero = 0;

#ifdef FAR_TABLES

        dict = farmalloc (MAX_DICT * sizeof (unsigned char));
        base_level = farmalloc (MAX_DICT * sizeof (unsigned char));

#ifndef SPLIT_TABLES

        next_dict = farmalloc ((MAX_DICT + HTBL1_SIZE) * sizeof (unsigned int));

        if (dict == NULL || base_level == NULL || next_dict == NULL)
        {
                printf ("Memory allocation failure\n");
                exit (EXIT_FAILURE);
        }

#else

        next_dict [0] = farmalloc ((MAX_DICT + HTBL1_SIZE + 1) / 2 * sizeof (unsigned int));
        next_dict [1] = farmalloc ((MAX_DICT + HTBL1_SIZE + 1) / 2 * sizeof (unsigned int));

        if (dict == NULL || base_level == NULL || next_dict [1] == NULL)
        {
                printf ("Memory allocation failure\n");
                exit (EXIT_FAILURE);
        }

        #endif  /* SPLIT_TABLES */

#endif  /* FAR_TABLES */

        for (i = 0; i < MAX_SYM; i ++)
        {
                freq_zero [i] = 1;
                order_zero [i] = 0;
        }

        for (i = 0; i < MAX_DICT + HTBL1_SIZE; i ++)
                NEXT_DICT (i) = NIL_DICT_PTR;

        for (i = 0; i < MAX_DICT; i ++)
        {
                dict [i] = 0;
                base_level [i] = 0;
        }

        save_state_index = 0;
        string_pos = 0;
        string_len = 0;
        string_start = 0;
        skip_count = MIN_STR;
        string_state = STRING_WAIT;
}




/*
        Compress next symbol
*/

void CompressSymbol (int ch)

{
        int i;
        unsigned cfreq;

        if (string_state == STRING_ACTIVE) check_string_cont ();

        if (dup_count > max_order + 2)
        {
                ++ active_state.order_cum_freq [max_order + 1];
                ++ freq [dup_char];

                if (ch != dup_char)
                {
                        for (i = 0, cfreq = 0; i < ch; i ++)
                        {
                                if (level [i] == level [ch]) cfreq += freq [i];
                        }
                        
                        base_freq [level [ch]] = cfreq;
                }
        }
        else
        {
                clear_context ();
                for (i = 0; i < max_order + 2; i ++) base_freq [i] = 0;
                scan_dict (ch);
        }

        for (i = 1; i <= active_state.initial_level; i ++)
        {
                while (active_state.order_cum_freq [i] > MAX_CUM_FREQ)
                        scale_freq_tbl (i, ch);
        }

        test_string_state ();

        calc_symbol_freq (ch);
        select_output_symbol ();
        update_model (ch);
}



/*
        Expand next symbol from input stream
*/

int ExpandSymbol (void)

{
        int ch;

        if (dup_count > max_order + 2)
        {
                ++ active_state.order_cum_freq [max_order + 1];
                ++ freq [dup_char];
        }
        else
        {
                clear_context ();
                scan_dict (0);
        }

        ch = string_len == 0 ? decode_symbol () : decode_string_char ();
        update_model (ch);
        
        return ch;
}



/*
        Update tables used by model for new symbol
        Link new symbol into dictionary
        Delete oldest symbol in dictionary, if required
        Update zero order frequencies if no higher level context found
*/

static void update_model (int ch)

{
        int n;

        NEXT_DICT (node) = NIL_DICT_PTR;
        NEXT_DICT (last_search_node) = node;
        last_search_node = node;

        if (active_state.match_level < 2)
        {
                cum_freq_zero ++;
                if (order_zero [ch])
                        freq_zero [ch] ++;
                else
                {
                        order_zero [ch] = 1;
                        sym_count_zero ++;
                }
        }

        dict [node] = ch;
        if (ch == dup_char)
                dup_count ++;
        else
        {
                dup_char = ch;
                dup_count = 0;
        }

        n = active_state.match_level;
        if (n == 0) n = 1;
        base_level [node] = n;

        delete_dict_entry (node + max_order);

        active_state.initial_level = active_state.match_level;
        if (active_state.initial_level <= max_order)
                active_state.initial_level ++;

        if (++ node == MAX_DICT)
        {
                node = MAX_ORDER;
                for (n = 1; n <= max_order; n ++)
                        dict [MAX_ORDER - n] = dict [MAX_DICT - n];
        }
}



/*
        Delete oldest symbol from dictionary
        Update frequency counts if added for lower order
*/

static void delete_dict_entry (int i)

{
        unsigned int n;
        int j;

        n = i;
        if (n >= MAX_DICT) n -= NDICT;

        switch (base_level [n])
        {
                case 0:
                        break;

                case 1:
                        j = dict [n];
                        cum_freq_zero --;
                        if (-- freq_zero [j] == 0)
                        {
                                order_zero [j] = 0;
                                freq_zero [j] = 1;
                                sym_count_zero --;
                        }

                default:
                        j = dict [n - 1];
                        NEXT_DICT (j + MAX_DICT) = NEXT_DICT (n);
                        break;
        }
}


/*
        Initialize tables at start of dictionary scan
        Establish counts based on low order tables
*/

static void clear_context (void)

{
        int i;

        for (i = 2; i < max_order + 2; i ++)
        {
                active_state.order_sym_count [i] = 0;
                active_state.order_cum_freq [i] = 0;
        }

        active_state.order_sym_count [1] = sym_count_zero;
        active_state.order_sym_count [0] = MAX_SYM - sym_count_zero;

        active_state.order_cum_freq [1] = cum_freq_zero;
        active_state.order_cum_freq [0] = MAX_SYM - sym_count_zero;

        movmem (freq_zero, freq, sizeof freq);
        movmem (order_zero, level, sizeof level);
}



/*
        Perform search of dictionary to locate active contexts
        Accumulate freqencies for all symbols encounered
        Use initial character of input string to select the starting table value
*/

static void scan_dict (int base_char)

{
        unsigned int k;
        unsigned int jnode;
        int ch;
        int n;

        last_search_node = dict [node - 1] + MAX_DICT;
        jnode = NEXT_DICT (last_search_node);

        while (jnode != NIL_DICT_PTR)
        {
                ch = dict [jnode];
                for (n = 2; n < max_order + 1; n ++)
                {
                        if (dict [node - n] != dict [jnode - n]) break;
                }

                switch (string_state)
                {
                        case STRING_SEARCH:
                        case STRING_START:
                                test_string_start (jnode, n);
                                break;

                        case STRING_COMP:
                                test_string_resume (jnode, n);
                                break;
                }

                if (base_level [jnode] <= n)
                {
                        k = level [ch];
                        if (k < n)
                        {
                                active_state.order_cum_freq [k] -= freq [ch];
                                active_state.order_sym_count [k] --;

                                active_state.order_cum_freq [n] ++;
                                active_state.order_sym_count [n] ++;

                                if (ch < base_char)
                                {
                                        base_freq [k] -= freq [ch];
                                        base_freq [n] ++;
                                }

                                level [ch] = n;
                                freq [ch] = 1;

                                if (n > active_state.initial_level) active_state.initial_level = n;
                        }
                        else
                        if (k == n)
                        {
                                active_state.order_cum_freq [n] ++;
                                freq [ch] ++;
                                if (ch < base_char) base_freq [n] ++;
                        }
                }

                last_search_node = jnode;
                jnode = NEXT_DICT (jnode);
        }
}



/*
        Test for continued text substring
        Executed before the start of each dictionary scan during compression
*/

static void check_string_cont (void)

{
        int j;

        j = string_pos + string_len;
        if (j >= MAX_DICT) j -= NDICT;

        if (dict [node - 1] == dict [j])
                string_len ++;
        else
                string_state = STRING_COMP;
}



/*
        Test for start of potential text substring
        Executed during dictionary scan based on string state variable
*/

static void test_string_start (unsigned pos, int n)

{
        if (n > MIN_STR)
        {
                string_pos = pos;
                if (string_pos < MIN_STR + MAX_ORDER) string_pos += NDICT;
                string_pos -= MIN_STR;
                string_len = MIN_STR;
                string_state = STRING_START;
        }
}



static void test_string_resume (unsigned pos, int n)

{
        if (n > string_len + 1)
        {
                string_state = STRING_ACTIVE;
                string_len ++;
                string_pos = pos;
                if (string_pos < string_len + MAX_ORDER) string_pos += NDICT;
                string_pos -= string_len;
        }
        else
        if (n == max_order + 1)
        {
                unsigned i2, j2, n2;

                i2 = node - max_order - 1;
                j2 = pos - max_order - 1;

                for (n2 = max_order + 1; n2 <= string_len + 1; n2 ++)
                {
                        if (i2 < MAX_ORDER) i2 += NDICT;
                        if (j2 < MAX_ORDER) j2 += NDICT;
                        if (dict [i2 --] != dict [j2 --]) break;
                }

                if (n2 > string_len + 1)
                {
                        string_state = STRING_ACTIVE;
                        string_len ++;
                        string_pos = pos;
                        if (string_pos < string_len + MAX_ORDER) string_pos += NDICT;
                        string_pos -= string_len;
                }
        }
}



/*
        Test status of text substring match procedure
        Performed after each dictionary scan
*/

static void test_string_state (void)

{
        switch (string_state)
        {
                case STRING_WAIT:
                        save_state [string_start].mod = active_state;
                        SaveCoderState (&save_state [string_start].cs);

                        if (++ string_start == MAX_STR_SAVE) string_start = 0;
                        if (-- skip_count == 0) string_state = STRING_SEARCH;
                        break;

                case STRING_SEARCH:
                        save_state [string_start].mod = active_state;
                        SaveCoderState (&save_state [string_start].cs);
                        if (++ string_start == MAX_STR_SAVE) string_start = 0;
                        break;

                case STRING_ACTIVE:
                        if (string_len > MAX_STR)
                        {
                                string_len --;
                                clear_text_string ();
                        }
                        break;

                case STRING_COMP:
                        clear_text_string ();
                        break;
        }
}



/*
        End of text substring
        Test for minimum code length of dtring versus coded symbols
        Reposition output and write string selection if less
        Set up for start of next string
*/

static void clear_text_string (void)

{
        int nbits;
        int i;

        if (string_len >= MIN_STR)
        {
                nbits = find_string_len ();
                if (nbits > 0)
                {
                        replace_text_string ();

                        i = node;
                        if (i <= string_len) i += NDICT;
                        i -= string_len + 1;
                }
        }

        save_state [string_start].mod = last_state;
        SaveCoderState (&save_state [string_start].cs);

        if (++ string_start == MAX_STR_SAVE) string_start = 0;

        encode_symbol (&last_state);

        save_state [string_start].mod = active_state;
        SaveCoderState (&save_state [string_start].cs);

        if (++ string_start == MAX_STR_SAVE) string_start = 0;
        skip_count = MIN_STR - 2;
        string_len = 0;

        string_state = STRING_WAIT;
}



/*
        Estimate size of string reference
        Returns size relative to actual length used by coded symbols
*/

static int find_string_len (void)

{
        int nbits;
        unsigned f;

        struct model start_context;
        unsigned i;
        unsigned n;
        unsigned j;
        int m;
        unsigned c1;

        nbits = 0;
        f = 0;

        new_symbol_count = MAX_SYM;
        m = save_state [string_start].mod.match_level;
        start_context = save_state [string_start].mod;

/*
        Accumulate switch characters for default context
        Include start of string symbol
*/

        do
        {
                new_symbol_count -= start_context.order_sym_count [m];

                c1 = start_context.order_cum_freq [m];
                n = switch_char_freq (start_context.order_sym_count [m], c1);

                update_bit_len (n, c1 + n, &nbits, &f);
        } while (-- m > 0);

        c1 = start_context.order_cum_freq [0];
        update_bit_len (1, c1, &nbits, &f);

/*
        Include string length
*/

        nbits += 6;
        if (string_len - MIN_STR >= 63) nbits += 8;

/*
        Calculate relative location of start of string
        Output as bit count and offset
*/

        update_bit_len (1, max_pos_size, &nbits, &f);

        i = string_pos + string_len;
        if (i >= MAX_DICT) i -= NDICT;
        j = i < node ? node - i - 1 : node + NDICT - 1 - i;
        nbits += log2 (j);

        return CodeLength (&save_state [string_start].cs) - nbits;
}



/*
        Generate coded symbols for text substring
        Used when string length is less length used by actual symbols
*/

static void replace_text_string (void)

{
        struct model start_context;
        unsigned n;
        unsigned i, j;

        RestoreCoderState (&save_state [string_start].cs);

        new_symbol_count = MAX_SYM;
        start_context = save_state [string_start].mod;

/*
        Output switch codes for default context and start string symbol
*/

        start_context.match_level = 0;
        start_context.base_count = start_context.order_cum_freq [0] - 1;
        start_context.sym_freq = 1;

        encode_symbol (&start_context);

/*
        Output string length
*/
        n = string_len - MIN_STR;
        if (n < MAX_STR_CODE - 1)
                generate_value (n, MAX_STR_CODE);
        else
        {
                generate_value (MAX_STR_CODE - 1, MAX_STR_CODE);
                generate_value (n + 1 - MAX_STR_CODE, 256);
        }

/*
        Determine relative location of start of string
*/

        i = string_pos + string_len;
        if (i >= MAX_DICT) i -= NDICT;
        j = i < node ? node - i - 1 : node - i - 1 + NDICT;
        n = 2;
        if (j < 2)
                i = 0;
        else
        {
                for (i = 1; 2 * n <= j && n < 0x8000; i ++, n <<= 1);
                j -= n;
        }

/*
        Output bit length of relative string location
*/

        generate_value (i, max_pos_size);

/*
        Output string location in 8 bit pices if required
*/

        if (i > 8)
        {
                generate_value (j & 0xFF, 256);
                j >>= 8;
                n >>= 8;
        }

        generate_value (j, n);
}


/*
        Scale frequency tables if total cumulative frequency exceeds maximum
        Also recalculates frequencies for current input symbol
*/

static void scale_freq_tbl (int order, int ch)

{
        int i;
        unsigned cfreq;
        unsigned t;

        i = 0;
        cfreq = 0;

        if (level [ch] == order)
        {
                for (; i < ch; i ++)
                {
                        if (level [i] == order)
                        {
                                t = (freq [i] + 1) >> 1;
                                freq [i] = t;
                                cfreq += t;
                        }
                }

                base_freq [order] = cfreq;

                t = (freq [i] + 1) >> 1;
                freq [i] = t;
                i ++;

                cfreq += t;
        }

        for (; i < MAX_CHAR_CODE; i ++)
        {
                if (level [i] == order)
                {
                        t = (freq [i] + 1) >> 1;
                        freq [i] = t;
                        cfreq += t;
                }
        }

        active_state.order_cum_freq [order] = cfreq;
}


/*
        Determine symbol frequency counts for coder
*/

static void calc_symbol_freq (int ch)

{
        int i;

        active_state.match_level = level [        active_state.sym_freq = freq [ch];

        if (active_state.match_level > 1)
                active_state.base_count = base_freq [active_state.match_level];
        else
        {
                active_state.base_count = 0;
                for (i = 0; i < ch; i ++)
                {
                        if (level [i] == active_state.match_level)
                                active_state.base_count += freq [i];
                }
        }
}


/*
        Select state structure containing symbol to be output
        Generate coded symbol value if required

        Note that there is a one character delay during substring matching
        so that the non matching character at end of string is not output
        until after string replacement has occurred
*/

static void select_output_symbol (void)

{
        switch (string_state)
        {
                case STRING_START:
                        last_state = active_state;
                        string_state = STRING_ACTIVE;
                        break;

                case STRING_ACTIVE:
                        encode_symbol (&last_state);
                        last_state = active_state;
                        break;

                default:
                        encode_symbol (&active_state);
                        break;
        }
}


/*
        Generate coded value for symbol defined by input state variable
        State includes symbol frequencies and all values used by coder
*/

static void encode_symbol (struct model *cptr)

{
        int i;

        new_symbol_count = MAX_SYM;
        i = cptr -> match_level;
        while (i < cptr -> initial_level)
        {
                new_symbol_count -= cptr -> order_sym_count [cptr -> initial_level];
                generate_switch_code (cptr);
                cptr -> initial_level --;
        }

        new_symbol_count -= cptr -> order_sym_count [i];
        generate_symbol_code (cptr);
}


/*
        Generate code for switch character to select next lower context
        Input consists of state variable for symbol
*/

static void generate_switch_code (struct model *cptr)

{
        unsigned c1;
        unsigned n, m;

        n = cptr -> initial_level;
        c1 = cptr -> order_cum_freq [n];
        m = switch_char_freq (cptr -> order_sym_count [n], c1);

        EncodeArith (c1, m, c1 + m);
}


/*
        Generate code for symbol defined by input state variable
*/

static void generate_symbol_code (struct model *cptr)

{
        unsigned int c1, c2, c3;
        int n;

        n = cptr -> initial_level;
        c1 = cptr -> base_count;
        c2 = cptr -> sym_freq;
        c3 = cptr -> order_cum_freq [n];

        if (n > 0) c3 += switch_char_freq (cptr -> order_sym_count [n], c3);

        EncodeArith (c1, c2, c3);
}



/*
        Estimate frequency to be allocated to the switch character
        Use number of symbols referenced in current context and the
        number of unreferenced symbols so far

        Value should reflect probability of encountering a symbol
        already present in the active context
*/

static unsigned switch_char_freq (unsigned scount, unsigned cmax)

{       unsigned n;

        n = (scount + 1) * new_symbol_count / (scount + new_symbol_count);
        if (n + cmax > MAX_CUM_FREQ) n = MAX_CUM_FREQ + 1 - cmax;

        return n;
}


/*
        End of compression procedure
        Terminate active substring processing
        Close arithmetic coder and flush output
*/

void CloseModel (void)

{
        if (string_state == STRING_ACTIVE) clear_text_string ();
        CloseCoder ();
}



/*
        Decode next input symbol from input stream
        Test for context switch and repeat until non switch symbol encountered
*/

static int decode_symbol (void)

{
        int ch;

        new_symbol_count = MAX_SYM;
        active_state.match_level = active_state.initial_level;

        new_symbol_count -= active_state.order_sym_count [active_state.match_level];
        ch = decode_active_state ();

        while (ch == SWITCH_SYM)
        {
                active_state.match_level --;
                new_symbol_count -= active_state.order_sym_count [active_state.match_level];
                ch = decode_active_state ();
        }

        if (ch == START_STRING) ch = start_decode_string ();

        return ch;
}



/*
        Start of string symbol encountered
        Extract string length and position from input stream
        Returns first character in string
*/

static int start_decode_string (void)

{
        unsigned i, j, k;
        unsigned n;

        string_len = decode_value (MAX_STR_CODE);
        if (string_len == MAX_STR_CODE - 1) string_len += decode_value (256);
        string_len += MIN_STR;

        i = decode_value (max_pos_size);
        if (i == 0)
        {
                i = 2;
                j = decode_value (2);
        }
        else
        if (i > 8)
        {
                j = decode_value (256);
                i -= 8;
                n = 1 << i;
                k = decode_value (n);
                k += n;
                k <<= 8;
                j += k;
        }
        else
        {
                n = 1 << i;
                j = decode_value (n);
                j += n;
        }

        string_pos = node < j + MAX_ORDER ? node + NDICT - j : node - j;

        return decode_string_char ();
}



/*
        Locate next character in text substring
        Increment string pointers and decrement length
        Set parameters used for updating state variable
*/

static int decode_string_char (void)

{
        int ch;
        unsigned int j;

        ch = dict [string_pos];
        active_state.match_level = level [ch];

        string_len --;
        if (++ string_pos == MAX_DICT) string_pos = MAX_ORDER;

        last_search_node = dict [node - 1] + MAX_DICT;
        j = NEXT_DICT (last_search_node);

        while (j != NIL_DICT_PTR)
        {
                last_search_node = j;
                j = NEXT_DICT (j);
        }

        return ch;
}


/*
        Extract next value from input stream
        Search frequency tables to convert to symbol
        Update arithmetic decoder using symbol frequencies
*/

static int decode_active_state (void)

{
        unsigned c1, c2, c3;
        unsigned sym;
        unsigned m;
        int i;
        int n;

        n = active_state.match_level;
        c2 = active_state.order_cum_freq [n];

        while (c2 > MAX_CUM_FREQ)
        {
                scale_freq_tbl (n, END_OF_FILE);
                c2 = active_state.order_cum_freq [n];
        }

        m = n > 0 ? switch_char_freq (active_state.order_sym_count [n], c2) : 0;
        c3 = c2 + m;

        sym = DecodeArith (c3);
        if (sym < c2)
        {
                c1 = 0;
                for (i = 0; i < MAX_SYM; i ++)
                {
                        if (level [i] == n)
                        {
                                m = freq [i];
                                c1 += m;
                                if (sym < c1) break;
                        }
                }

                c1 -= m;
        }
        else
        {
                c1 = c2;
                i = SWITCH_SYM;
        }

        UpdateDecoder (c1, m, c3);

        return i;
}



/*
        Generate coded output for a constant value within a fixed range
        Value is treated as a symbol with frequency one
*/

static void generate_value (unsigned value, unsigned range)

{
        EncodeArith (value, 1, range);
}


/*
        Extract constant value within fixed range
        Each possible value is treated as symbol with frequency one
*/

static unsigned decode_value (unsigned range)

{       unsigned value;

        value = DecodeArith (range);
        UpdateDecoder (value, 1, range);

        return value;
}



/*
        Determine integer value of base 2 logarithm of integer value
        Use smallest power of two less than input value
*/

static int log2 (unsigned n)

{
        int i;

        for (i = 0; n > 1; i ++, n >>= 1);
        return i;
}



/*
        Scale binary value as part of log calulation
        Input is a binary fraction (32 bits, 16 binary places)
        Extract integer log to base 2
        Return fractional part and accumulate integer part
*/

static void scale_binary_value (long x, int *nbits, unsigned *frac)

{
        int i;

        i = log2 ((unsigned) (x >> 16));
        *nbits += i;
        *frac = (unsigned) (x >> i);
}


/*
        Estimate bit size of arithmetic coded values

        Input consists of symbol frequency (p) and cumulative frequency (q)
        Actual bit size is log to base 2 of (q / p)
        
        Maintain intermediate values as:
        
                n + log2 (1 + f)

        where n is integer and f is the remainder (between 0 and 1.0)
        stored as a 16 bit binary fraction.

        The pair (nbits, frac) contains the starting value for the bit length
        This is updated to include the latest symbol size
*/

static void update_bit_len (unsigned p, unsigned q, int *nbits, unsigned *frac)

{
        long x;
        unsigned f2;

        x = ((long) q << 16) / p;
        scale_binary_value (x, nbits, &f2);

        x = (long) f2 * (long) *frac;
        x >>= 16;
        x += f2;
        x += *frac;
        x += 0x10000L;
        scale_binary_value (x, nbits, frac);
}

--

Every problem has a solution.
                                infosite@263.net
※ 修改:.haojs 于 Sep  3 08:23:48 修改本文.[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)
页面执行时间:604.548毫秒