annotate src/bzip2-1.0.6/huffman.c @ 23:619f715526df sv_v2.1

Update Vamp plugin SDK to 2.5
author Chris Cannam
date Thu, 09 May 2013 10:52:46 +0100
parents e13257ea84a4
children
rev   line source
Chris@4 1
Chris@4 2 /*-------------------------------------------------------------*/
Chris@4 3 /*--- Huffman coding low-level stuff ---*/
Chris@4 4 /*--- huffman.c ---*/
Chris@4 5 /*-------------------------------------------------------------*/
Chris@4 6
Chris@4 7 /* ------------------------------------------------------------------
Chris@4 8 This file is part of bzip2/libbzip2, a program and library for
Chris@4 9 lossless, block-sorting data compression.
Chris@4 10
Chris@4 11 bzip2/libbzip2 version 1.0.6 of 6 September 2010
Chris@4 12 Copyright (C) 1996-2010 Julian Seward <jseward@bzip.org>
Chris@4 13
Chris@4 14 Please read the WARNING, DISCLAIMER and PATENTS sections in the
Chris@4 15 README file.
Chris@4 16
Chris@4 17 This program is released under the terms of the license contained
Chris@4 18 in the file LICENSE.
Chris@4 19 ------------------------------------------------------------------ */
Chris@4 20
Chris@4 21
Chris@4 22 #include "bzlib_private.h"
Chris@4 23
Chris@4 24 /*---------------------------------------------------*/
Chris@4 25 #define WEIGHTOF(zz0) ((zz0) & 0xffffff00)
Chris@4 26 #define DEPTHOF(zz1) ((zz1) & 0x000000ff)
Chris@4 27 #define MYMAX(zz2,zz3) ((zz2) > (zz3) ? (zz2) : (zz3))
Chris@4 28
Chris@4 29 #define ADDWEIGHTS(zw1,zw2) \
Chris@4 30 (WEIGHTOF(zw1)+WEIGHTOF(zw2)) | \
Chris@4 31 (1 + MYMAX(DEPTHOF(zw1),DEPTHOF(zw2)))
Chris@4 32
Chris@4 33 #define UPHEAP(z) \
Chris@4 34 { \
Chris@4 35 Int32 zz, tmp; \
Chris@4 36 zz = z; tmp = heap[zz]; \
Chris@4 37 while (weight[tmp] < weight[heap[zz >> 1]]) { \
Chris@4 38 heap[zz] = heap[zz >> 1]; \
Chris@4 39 zz >>= 1; \
Chris@4 40 } \
Chris@4 41 heap[zz] = tmp; \
Chris@4 42 }
Chris@4 43
Chris@4 44 #define DOWNHEAP(z) \
Chris@4 45 { \
Chris@4 46 Int32 zz, yy, tmp; \
Chris@4 47 zz = z; tmp = heap[zz]; \
Chris@4 48 while (True) { \
Chris@4 49 yy = zz << 1; \
Chris@4 50 if (yy > nHeap) break; \
Chris@4 51 if (yy < nHeap && \
Chris@4 52 weight[heap[yy+1]] < weight[heap[yy]]) \
Chris@4 53 yy++; \
Chris@4 54 if (weight[tmp] < weight[heap[yy]]) break; \
Chris@4 55 heap[zz] = heap[yy]; \
Chris@4 56 zz = yy; \
Chris@4 57 } \
Chris@4 58 heap[zz] = tmp; \
Chris@4 59 }
Chris@4 60
Chris@4 61
Chris@4 62 /*---------------------------------------------------*/
Chris@4 63 void BZ2_hbMakeCodeLengths ( UChar *len,
Chris@4 64 Int32 *freq,
Chris@4 65 Int32 alphaSize,
Chris@4 66 Int32 maxLen )
Chris@4 67 {
Chris@4 68 /*--
Chris@4 69 Nodes and heap entries run from 1. Entry 0
Chris@4 70 for both the heap and nodes is a sentinel.
Chris@4 71 --*/
Chris@4 72 Int32 nNodes, nHeap, n1, n2, i, j, k;
Chris@4 73 Bool tooLong;
Chris@4 74
Chris@4 75 Int32 heap [ BZ_MAX_ALPHA_SIZE + 2 ];
Chris@4 76 Int32 weight [ BZ_MAX_ALPHA_SIZE * 2 ];
Chris@4 77 Int32 parent [ BZ_MAX_ALPHA_SIZE * 2 ];
Chris@4 78
Chris@4 79 for (i = 0; i < alphaSize; i++)
Chris@4 80 weight[i+1] = (freq[i] == 0 ? 1 : freq[i]) << 8;
Chris@4 81
Chris@4 82 while (True) {
Chris@4 83
Chris@4 84 nNodes = alphaSize;
Chris@4 85 nHeap = 0;
Chris@4 86
Chris@4 87 heap[0] = 0;
Chris@4 88 weight[0] = 0;
Chris@4 89 parent[0] = -2;
Chris@4 90
Chris@4 91 for (i = 1; i <= alphaSize; i++) {
Chris@4 92 parent[i] = -1;
Chris@4 93 nHeap++;
Chris@4 94 heap[nHeap] = i;
Chris@4 95 UPHEAP(nHeap);
Chris@4 96 }
Chris@4 97
Chris@4 98 AssertH( nHeap < (BZ_MAX_ALPHA_SIZE+2), 2001 );
Chris@4 99
Chris@4 100 while (nHeap > 1) {
Chris@4 101 n1 = heap[1]; heap[1] = heap[nHeap]; nHeap--; DOWNHEAP(1);
Chris@4 102 n2 = heap[1]; heap[1] = heap[nHeap]; nHeap--; DOWNHEAP(1);
Chris@4 103 nNodes++;
Chris@4 104 parent[n1] = parent[n2] = nNodes;
Chris@4 105 weight[nNodes] = ADDWEIGHTS(weight[n1], weight[n2]);
Chris@4 106 parent[nNodes] = -1;
Chris@4 107 nHeap++;
Chris@4 108 heap[nHeap] = nNodes;
Chris@4 109 UPHEAP(nHeap);
Chris@4 110 }
Chris@4 111
Chris@4 112 AssertH( nNodes < (BZ_MAX_ALPHA_SIZE * 2), 2002 );
Chris@4 113
Chris@4 114 tooLong = False;
Chris@4 115 for (i = 1; i <= alphaSize; i++) {
Chris@4 116 j = 0;
Chris@4 117 k = i;
Chris@4 118 while (parent[k] >= 0) { k = parent[k]; j++; }
Chris@4 119 len[i-1] = j;
Chris@4 120 if (j > maxLen) tooLong = True;
Chris@4 121 }
Chris@4 122
Chris@4 123 if (! tooLong) break;
Chris@4 124
Chris@4 125 /* 17 Oct 04: keep-going condition for the following loop used
Chris@4 126 to be 'i < alphaSize', which missed the last element,
Chris@4 127 theoretically leading to the possibility of the compressor
Chris@4 128 looping. However, this count-scaling step is only needed if
Chris@4 129 one of the generated Huffman code words is longer than
Chris@4 130 maxLen, which up to and including version 1.0.2 was 20 bits,
Chris@4 131 which is extremely unlikely. In version 1.0.3 maxLen was
Chris@4 132 changed to 17 bits, which has minimal effect on compression
Chris@4 133 ratio, but does mean this scaling step is used from time to
Chris@4 134 time, enough to verify that it works.
Chris@4 135
Chris@4 136 This means that bzip2-1.0.3 and later will only produce
Chris@4 137 Huffman codes with a maximum length of 17 bits. However, in
Chris@4 138 order to preserve backwards compatibility with bitstreams
Chris@4 139 produced by versions pre-1.0.3, the decompressor must still
Chris@4 140 handle lengths of up to 20. */
Chris@4 141
Chris@4 142 for (i = 1; i <= alphaSize; i++) {
Chris@4 143 j = weight[i] >> 8;
Chris@4 144 j = 1 + (j / 2);
Chris@4 145 weight[i] = j << 8;
Chris@4 146 }
Chris@4 147 }
Chris@4 148 }
Chris@4 149
Chris@4 150
Chris@4 151 /*---------------------------------------------------*/
Chris@4 152 void BZ2_hbAssignCodes ( Int32 *code,
Chris@4 153 UChar *length,
Chris@4 154 Int32 minLen,
Chris@4 155 Int32 maxLen,
Chris@4 156 Int32 alphaSize )
Chris@4 157 {
Chris@4 158 Int32 n, vec, i;
Chris@4 159
Chris@4 160 vec = 0;
Chris@4 161 for (n = minLen; n <= maxLen; n++) {
Chris@4 162 for (i = 0; i < alphaSize; i++)
Chris@4 163 if (length[i] == n) { code[i] = vec; vec++; };
Chris@4 164 vec <<= 1;
Chris@4 165 }
Chris@4 166 }
Chris@4 167
Chris@4 168
Chris@4 169 /*---------------------------------------------------*/
Chris@4 170 void BZ2_hbCreateDecodeTables ( Int32 *limit,
Chris@4 171 Int32 *base,
Chris@4 172 Int32 *perm,
Chris@4 173 UChar *length,
Chris@4 174 Int32 minLen,
Chris@4 175 Int32 maxLen,
Chris@4 176 Int32 alphaSize )
Chris@4 177 {
Chris@4 178 Int32 pp, i, j, vec;
Chris@4 179
Chris@4 180 pp = 0;
Chris@4 181 for (i = minLen; i <= maxLen; i++)
Chris@4 182 for (j = 0; j < alphaSize; j++)
Chris@4 183 if (length[j] == i) { perm[pp] = j; pp++; };
Chris@4 184
Chris@4 185 for (i = 0; i < BZ_MAX_CODE_LEN; i++) base[i] = 0;
Chris@4 186 for (i = 0; i < alphaSize; i++) base[length[i]+1]++;
Chris@4 187
Chris@4 188 for (i = 1; i < BZ_MAX_CODE_LEN; i++) base[i] += base[i-1];
Chris@4 189
Chris@4 190 for (i = 0; i < BZ_MAX_CODE_LEN; i++) limit[i] = 0;
Chris@4 191 vec = 0;
Chris@4 192
Chris@4 193 for (i = minLen; i <= maxLen; i++) {
Chris@4 194 vec += (base[i+1] - base[i]);
Chris@4 195 limit[i] = vec-1;
Chris@4 196 vec <<= 1;
Chris@4 197 }
Chris@4 198 for (i = minLen + 1; i <= maxLen; i++)
Chris@4 199 base[i] = ((limit[i-1] + 1) << 1) - base[i];
Chris@4 200 }
Chris@4 201
Chris@4 202
Chris@4 203 /*-------------------------------------------------------------*/
Chris@4 204 /*--- end huffman.c ---*/
Chris@4 205 /*-------------------------------------------------------------*/