crc32.c revision 1.3 1 1.3 uwe /* $NetBSD: crc32.c,v 1.3 2006/01/25 23:15:55 uwe Exp $ */
2 1.1 christos
3 1.1 christos /* crc32.c -- compute the CRC-32 of a data stream
4 1.1 christos * Copyright (C) 1995-2005 Mark Adler
5 1.1 christos * For conditions of distribution and use, see copyright notice in zlib.h
6 1.1 christos *
7 1.1 christos * Thanks to Rodney Brown <rbrown64 (at) csc.com.au> for his contribution of faster
8 1.1 christos * CRC methods: exclusive-oring 32 bits of data at a time, and pre-computing
9 1.1 christos * tables for updating the shift register in one step with three exclusive-ors
10 1.1 christos * instead of four steps with four exclusive-ors. This results in about a
11 1.1 christos * factor of two increase in speed on a Power PC G4 (PPC7455) using gcc -O3.
12 1.1 christos */
13 1.1 christos
14 1.1 christos /* @(#) Id */
15 1.1 christos
16 1.1 christos /*
17 1.1 christos Note on the use of DYNAMIC_CRC_TABLE: there is no mutex or semaphore
18 1.1 christos protection on the static variables used to control the first-use generation
19 1.1 christos of the crc tables. Therefore, if you #define DYNAMIC_CRC_TABLE, you should
20 1.1 christos first call get_crc_table() to initialize the tables before allowing more than
21 1.1 christos one thread to use crc32().
22 1.1 christos */
23 1.1 christos
24 1.1 christos #ifdef MAKECRCH
25 1.1 christos # include <stdio.h>
26 1.1 christos # ifndef DYNAMIC_CRC_TABLE
27 1.1 christos # define DYNAMIC_CRC_TABLE
28 1.1 christos # endif /* !DYNAMIC_CRC_TABLE */
29 1.1 christos #endif /* MAKECRCH */
30 1.1 christos
31 1.1 christos #include "zutil.h" /* for STDC and FAR definitions */
32 1.1 christos
33 1.1 christos #define local static
34 1.1 christos
35 1.1 christos /* Find a four-byte integer type for crc32_little() and crc32_big(). */
36 1.1 christos #ifndef NOBYFOUR
37 1.3 uwe #if defined(__NetBSD__) && (defined(_KERNEL) || defined(_STANDALONE))
38 1.3 uwe # define BYFOUR
39 1.3 uwe typedef uint32_t u4;
40 1.3 uwe #else
41 1.1 christos # ifdef STDC /* need ANSI C limits.h to determine sizes */
42 1.1 christos # include <limits.h>
43 1.1 christos # define BYFOUR
44 1.1 christos # if (UINT_MAX == 0xffffffffUL)
45 1.1 christos typedef unsigned int u4;
46 1.1 christos # else
47 1.1 christos # if (ULONG_MAX == 0xffffffffUL)
48 1.1 christos typedef unsigned long u4;
49 1.1 christos # else
50 1.1 christos # if (USHRT_MAX == 0xffffffffUL)
51 1.1 christos typedef unsigned short u4;
52 1.1 christos # else
53 1.1 christos # undef BYFOUR /* can't find a four-byte integer type! */
54 1.1 christos # endif
55 1.1 christos # endif
56 1.1 christos # endif
57 1.1 christos # endif /* STDC */
58 1.3 uwe #endif /* __NetBSD__ && (_KERNEL || _STANDALONE) */
59 1.1 christos #endif /* !NOBYFOUR */
60 1.1 christos
61 1.1 christos /* Definitions for doing the crc four data bytes at a time. */
62 1.1 christos #ifdef BYFOUR
63 1.1 christos # define REV(w) (((w)>>24)+(((w)>>8)&0xff00)+ \
64 1.1 christos (((w)&0xff00)<<8)+(((w)&0xff)<<24))
65 1.1 christos local unsigned long crc32_little OF((unsigned long,
66 1.1 christos const unsigned char FAR *, unsigned));
67 1.1 christos local unsigned long crc32_big OF((unsigned long,
68 1.1 christos const unsigned char FAR *, unsigned));
69 1.1 christos # define TBLS 8
70 1.1 christos #else
71 1.1 christos # define TBLS 1
72 1.1 christos #endif /* BYFOUR */
73 1.1 christos
74 1.1 christos /* Local functions for crc concatenation */
75 1.1 christos local unsigned long gf2_matrix_times OF((unsigned long *mat,
76 1.1 christos unsigned long vec));
77 1.1 christos local void gf2_matrix_square OF((unsigned long *square, unsigned long *mat));
78 1.1 christos
79 1.1 christos #ifdef DYNAMIC_CRC_TABLE
80 1.1 christos
81 1.1 christos local volatile int crc_table_empty = 1;
82 1.1 christos local unsigned long FAR crc_table[TBLS][256];
83 1.1 christos local void make_crc_table OF((void));
84 1.1 christos #ifdef MAKECRCH
85 1.1 christos local void write_table OF((FILE *, const unsigned long FAR *));
86 1.1 christos #endif /* MAKECRCH */
87 1.1 christos /*
88 1.1 christos Generate tables for a byte-wise 32-bit CRC calculation on the polynomial:
89 1.1 christos x^32+x^26+x^23+x^22+x^16+x^12+x^11+x^10+x^8+x^7+x^5+x^4+x^2+x+1.
90 1.1 christos
91 1.1 christos Polynomials over GF(2) are represented in binary, one bit per coefficient,
92 1.1 christos with the lowest powers in the most significant bit. Then adding polynomials
93 1.1 christos is just exclusive-or, and multiplying a polynomial by x is a right shift by
94 1.1 christos one. If we call the above polynomial p, and represent a byte as the
95 1.1 christos polynomial q, also with the lowest power in the most significant bit (so the
96 1.1 christos byte 0xb1 is the polynomial x^7+x^3+x+1), then the CRC is (q*x^32) mod p,
97 1.1 christos where a mod b means the remainder after dividing a by b.
98 1.1 christos
99 1.1 christos This calculation is done using the shift-register method of multiplying and
100 1.1 christos taking the remainder. The register is initialized to zero, and for each
101 1.1 christos incoming bit, x^32 is added mod p to the register if the bit is a one (where
102 1.1 christos x^32 mod p is p+x^32 = x^26+...+1), and the register is multiplied mod p by
103 1.1 christos x (which is shifting right by one and adding x^32 mod p if the bit shifted
104 1.1 christos out is a one). We start with the highest power (least significant bit) of
105 1.1 christos q and repeat for all eight bits of q.
106 1.1 christos
107 1.1 christos The first table is simply the CRC of all possible eight bit values. This is
108 1.1 christos all the information needed to generate CRCs on data a byte at a time for all
109 1.1 christos combinations of CRC register values and incoming bytes. The remaining tables
110 1.1 christos allow for word-at-a-time CRC calculation for both big-endian and little-
111 1.1 christos endian machines, where a word is four bytes.
112 1.1 christos */
113 1.1 christos local void make_crc_table()
114 1.1 christos {
115 1.1 christos unsigned long c;
116 1.1 christos int n, k;
117 1.1 christos unsigned long poly; /* polynomial exclusive-or pattern */
118 1.1 christos /* terms of polynomial defining this crc (except x^32): */
119 1.1 christos static volatile int first = 1; /* flag to limit concurrent making */
120 1.1 christos static const unsigned char p[] = {0,1,2,4,5,7,8,10,11,12,16,22,23,26};
121 1.1 christos
122 1.1 christos /* See if another task is already doing this (not thread-safe, but better
123 1.1 christos than nothing -- significantly reduces duration of vulnerability in
124 1.1 christos case the advice about DYNAMIC_CRC_TABLE is ignored) */
125 1.1 christos if (first) {
126 1.1 christos first = 0;
127 1.1 christos
128 1.1 christos /* make exclusive-or pattern from polynomial (0xedb88320UL) */
129 1.1 christos poly = 0UL;
130 1.1 christos for (n = 0; n < sizeof(p)/sizeof(unsigned char); n++)
131 1.1 christos poly |= 1UL << (31 - p[n]);
132 1.1 christos
133 1.1 christos /* generate a crc for every 8-bit value */
134 1.1 christos for (n = 0; n < 256; n++) {
135 1.1 christos c = (unsigned long)n;
136 1.1 christos for (k = 0; k < 8; k++)
137 1.1 christos c = c & 1 ? poly ^ (c >> 1) : c >> 1;
138 1.1 christos crc_table[0][n] = c;
139 1.1 christos }
140 1.1 christos
141 1.1 christos #ifdef BYFOUR
142 1.1 christos /* generate crc for each value followed by one, two, and three zeros,
143 1.1 christos and then the byte reversal of those as well as the first table */
144 1.1 christos for (n = 0; n < 256; n++) {
145 1.1 christos c = crc_table[0][n];
146 1.1 christos crc_table[4][n] = REV(c);
147 1.1 christos for (k = 1; k < 4; k++) {
148 1.1 christos c = crc_table[0][c & 0xff] ^ (c >> 8);
149 1.1 christos crc_table[k][n] = c;
150 1.1 christos crc_table[k + 4][n] = REV(c);
151 1.1 christos }
152 1.1 christos }
153 1.1 christos #endif /* BYFOUR */
154 1.1 christos
155 1.1 christos crc_table_empty = 0;
156 1.1 christos }
157 1.1 christos else { /* not first */
158 1.1 christos /* wait for the other guy to finish (not efficient, but rare) */
159 1.1 christos while (crc_table_empty)
160 1.1 christos ;
161 1.1 christos }
162 1.1 christos
163 1.1 christos #ifdef MAKECRCH
164 1.1 christos /* write out CRC tables to crc32.h */
165 1.1 christos {
166 1.1 christos FILE *out;
167 1.1 christos
168 1.1 christos out = fopen("crc32.h", "w");
169 1.1 christos if (out == NULL) return;
170 1.1 christos fprintf(out, "/* crc32.h -- tables for rapid CRC calculation\n");
171 1.1 christos fprintf(out, " * Generated automatically by crc32.c\n */\n\n");
172 1.1 christos fprintf(out, "local const unsigned long FAR ");
173 1.1 christos fprintf(out, "crc_table[TBLS][256] =\n{\n {\n");
174 1.1 christos write_table(out, crc_table[0]);
175 1.1 christos # ifdef BYFOUR
176 1.1 christos fprintf(out, "#ifdef BYFOUR\n");
177 1.1 christos for (k = 1; k < 8; k++) {
178 1.1 christos fprintf(out, " },\n {\n");
179 1.1 christos write_table(out, crc_table[k]);
180 1.1 christos }
181 1.1 christos fprintf(out, "#endif\n");
182 1.1 christos # endif /* BYFOUR */
183 1.1 christos fprintf(out, " }\n};\n");
184 1.1 christos fclose(out);
185 1.1 christos }
186 1.1 christos #endif /* MAKECRCH */
187 1.1 christos }
188 1.1 christos
189 1.1 christos #ifdef MAKECRCH
190 1.1 christos local void write_table(out, table)
191 1.1 christos FILE *out;
192 1.1 christos const unsigned long FAR *table;
193 1.1 christos {
194 1.1 christos int n;
195 1.1 christos
196 1.1 christos for (n = 0; n < 256; n++)
197 1.1 christos fprintf(out, "%s0x%08lxUL%s", n % 5 ? "" : " ", table[n],
198 1.1 christos n == 255 ? "\n" : (n % 5 == 4 ? ",\n" : ", "));
199 1.1 christos }
200 1.1 christos #endif /* MAKECRCH */
201 1.1 christos
202 1.1 christos #else /* !DYNAMIC_CRC_TABLE */
203 1.1 christos /* ========================================================================
204 1.1 christos * Tables of CRC-32s of all single-byte values, made by make_crc_table().
205 1.1 christos */
206 1.1 christos #include "crc32.h"
207 1.1 christos #endif /* DYNAMIC_CRC_TABLE */
208 1.1 christos
209 1.1 christos /* =========================================================================
210 1.1 christos * This function can be used by asm versions of crc32()
211 1.1 christos */
212 1.1 christos const unsigned long FAR * ZEXPORT get_crc_table()
213 1.1 christos {
214 1.1 christos #ifdef DYNAMIC_CRC_TABLE
215 1.1 christos if (crc_table_empty)
216 1.1 christos make_crc_table();
217 1.1 christos #endif /* DYNAMIC_CRC_TABLE */
218 1.1 christos return (const unsigned long FAR *)crc_table;
219 1.1 christos }
220 1.1 christos
221 1.1 christos /* ========================================================================= */
222 1.1 christos #define DO1 crc = crc_table[0][((int)crc ^ (*buf++)) & 0xff] ^ (crc >> 8)
223 1.1 christos #define DO8 DO1; DO1; DO1; DO1; DO1; DO1; DO1; DO1
224 1.1 christos
225 1.1 christos /* ========================================================================= */
226 1.1 christos unsigned long ZEXPORT crc32(crc, buf, len)
227 1.1 christos unsigned long crc;
228 1.1 christos const unsigned char FAR *buf;
229 1.1 christos unsigned len;
230 1.1 christos {
231 1.1 christos if (buf == Z_NULL) return 0UL;
232 1.1 christos
233 1.1 christos #ifdef DYNAMIC_CRC_TABLE
234 1.1 christos if (crc_table_empty)
235 1.1 christos make_crc_table();
236 1.1 christos #endif /* DYNAMIC_CRC_TABLE */
237 1.1 christos
238 1.1 christos #ifdef BYFOUR
239 1.2 christos if (sizeof(void *) == sizeof(z_ptrdiff_t)) {
240 1.1 christos u4 endian;
241 1.1 christos
242 1.1 christos endian = 1;
243 1.1 christos if (*((unsigned char *)(&endian)))
244 1.1 christos return crc32_little(crc, buf, len);
245 1.1 christos else
246 1.1 christos return crc32_big(crc, buf, len);
247 1.1 christos }
248 1.1 christos #endif /* BYFOUR */
249 1.1 christos crc = crc ^ 0xffffffffUL;
250 1.1 christos while (len >= 8) {
251 1.1 christos DO8;
252 1.1 christos len -= 8;
253 1.1 christos }
254 1.1 christos if (len) do {
255 1.1 christos DO1;
256 1.1 christos } while (--len);
257 1.1 christos return crc ^ 0xffffffffUL;
258 1.1 christos }
259 1.1 christos
260 1.1 christos #ifdef BYFOUR
261 1.1 christos
262 1.1 christos /* ========================================================================= */
263 1.1 christos #define DOLIT4 c ^= *buf4++; \
264 1.1 christos c = crc_table[3][c & 0xff] ^ crc_table[2][(c >> 8) & 0xff] ^ \
265 1.1 christos crc_table[1][(c >> 16) & 0xff] ^ crc_table[0][c >> 24]
266 1.1 christos #define DOLIT32 DOLIT4; DOLIT4; DOLIT4; DOLIT4; DOLIT4; DOLIT4; DOLIT4; DOLIT4
267 1.1 christos
268 1.1 christos /* ========================================================================= */
269 1.1 christos local unsigned long crc32_little(crc, buf, len)
270 1.1 christos unsigned long crc;
271 1.1 christos const unsigned char FAR *buf;
272 1.1 christos unsigned len;
273 1.1 christos {
274 1.1 christos register u4 c;
275 1.1 christos register const u4 FAR *buf4;
276 1.1 christos
277 1.1 christos c = (u4)crc;
278 1.1 christos c = ~c;
279 1.2 christos while (len && ((z_ptrdiff_t)buf & 3)) {
280 1.1 christos c = crc_table[0][(c ^ *buf++) & 0xff] ^ (c >> 8);
281 1.1 christos len--;
282 1.1 christos }
283 1.1 christos
284 1.1 christos buf4 = (const u4 FAR *)(const void FAR *)buf;
285 1.1 christos while (len >= 32) {
286 1.1 christos DOLIT32;
287 1.1 christos len -= 32;
288 1.1 christos }
289 1.1 christos while (len >= 4) {
290 1.1 christos DOLIT4;
291 1.1 christos len -= 4;
292 1.1 christos }
293 1.1 christos buf = (const unsigned char FAR *)buf4;
294 1.1 christos
295 1.1 christos if (len) do {
296 1.1 christos c = crc_table[0][(c ^ *buf++) & 0xff] ^ (c >> 8);
297 1.1 christos } while (--len);
298 1.1 christos c = ~c;
299 1.1 christos return (unsigned long)c;
300 1.1 christos }
301 1.1 christos
302 1.1 christos /* ========================================================================= */
303 1.1 christos #define DOBIG4 c ^= *++buf4; \
304 1.1 christos c = crc_table[4][c & 0xff] ^ crc_table[5][(c >> 8) & 0xff] ^ \
305 1.1 christos crc_table[6][(c >> 16) & 0xff] ^ crc_table[7][c >> 24]
306 1.1 christos #define DOBIG32 DOBIG4; DOBIG4; DOBIG4; DOBIG4; DOBIG4; DOBIG4; DOBIG4; DOBIG4
307 1.1 christos
308 1.1 christos /* ========================================================================= */
309 1.1 christos local unsigned long crc32_big(crc, buf, len)
310 1.1 christos unsigned long crc;
311 1.1 christos const unsigned char FAR *buf;
312 1.1 christos unsigned len;
313 1.1 christos {
314 1.1 christos register u4 c;
315 1.1 christos register const u4 FAR *buf4;
316 1.1 christos
317 1.1 christos c = REV((u4)crc);
318 1.1 christos c = ~c;
319 1.2 christos while (len && ((z_ptrdiff_t)buf & 3)) {
320 1.1 christos c = crc_table[4][(c >> 24) ^ *buf++] ^ (c << 8);
321 1.1 christos len--;
322 1.1 christos }
323 1.1 christos
324 1.1 christos buf4 = (const u4 FAR *)(const void FAR *)buf;
325 1.1 christos buf4--;
326 1.1 christos while (len >= 32) {
327 1.1 christos DOBIG32;
328 1.1 christos len -= 32;
329 1.1 christos }
330 1.1 christos while (len >= 4) {
331 1.1 christos DOBIG4;
332 1.1 christos len -= 4;
333 1.1 christos }
334 1.1 christos buf4++;
335 1.1 christos buf = (const unsigned char FAR *)buf4;
336 1.1 christos
337 1.1 christos if (len) do {
338 1.1 christos c = crc_table[4][(c >> 24) ^ *buf++] ^ (c << 8);
339 1.1 christos } while (--len);
340 1.1 christos c = ~c;
341 1.1 christos return (unsigned long)(REV(c));
342 1.1 christos }
343 1.1 christos
344 1.1 christos #endif /* BYFOUR */
345 1.1 christos
346 1.1 christos #define GF2_DIM 32 /* dimension of GF(2) vectors (length of CRC) */
347 1.1 christos
348 1.1 christos /* ========================================================================= */
349 1.1 christos local unsigned long gf2_matrix_times(mat, vec)
350 1.1 christos unsigned long *mat;
351 1.1 christos unsigned long vec;
352 1.1 christos {
353 1.1 christos unsigned long sum;
354 1.1 christos
355 1.1 christos sum = 0;
356 1.1 christos while (vec) {
357 1.1 christos if (vec & 1)
358 1.1 christos sum ^= *mat;
359 1.1 christos vec >>= 1;
360 1.1 christos mat++;
361 1.1 christos }
362 1.1 christos return sum;
363 1.1 christos }
364 1.1 christos
365 1.1 christos /* ========================================================================= */
366 1.1 christos local void gf2_matrix_square(square, mat)
367 1.1 christos unsigned long *square;
368 1.1 christos unsigned long *mat;
369 1.1 christos {
370 1.1 christos int n;
371 1.1 christos
372 1.1 christos for (n = 0; n < GF2_DIM; n++)
373 1.1 christos square[n] = gf2_matrix_times(mat, mat[n]);
374 1.1 christos }
375 1.1 christos
376 1.1 christos /* ========================================================================= */
377 1.1 christos uLong ZEXPORT crc32_combine(crc1, crc2, len2)
378 1.1 christos uLong crc1;
379 1.1 christos uLong crc2;
380 1.1 christos z_off_t len2;
381 1.1 christos {
382 1.1 christos int n;
383 1.1 christos unsigned long row;
384 1.1 christos unsigned long even[GF2_DIM]; /* even-power-of-two zeros operator */
385 1.1 christos unsigned long odd[GF2_DIM]; /* odd-power-of-two zeros operator */
386 1.1 christos
387 1.1 christos /* degenerate case */
388 1.1 christos if (len2 == 0)
389 1.1 christos return crc1;
390 1.1 christos
391 1.1 christos /* put operator for one zero bit in odd */
392 1.1 christos odd[0] = 0xedb88320L; /* CRC-32 polynomial */
393 1.1 christos row = 1;
394 1.1 christos for (n = 1; n < GF2_DIM; n++) {
395 1.1 christos odd[n] = row;
396 1.1 christos row <<= 1;
397 1.1 christos }
398 1.1 christos
399 1.1 christos /* put operator for two zero bits in even */
400 1.1 christos gf2_matrix_square(even, odd);
401 1.1 christos
402 1.1 christos /* put operator for four zero bits in odd */
403 1.1 christos gf2_matrix_square(odd, even);
404 1.1 christos
405 1.1 christos /* apply len2 zeros to crc1 (first square will put the operator for one
406 1.1 christos zero byte, eight zero bits, in even) */
407 1.1 christos do {
408 1.1 christos /* apply zeros operator for this bit of len2 */
409 1.1 christos gf2_matrix_square(even, odd);
410 1.1 christos if (len2 & 1)
411 1.1 christos crc1 = gf2_matrix_times(even, crc1);
412 1.1 christos len2 >>= 1;
413 1.1 christos
414 1.1 christos /* if no more bits set, then done */
415 1.1 christos if (len2 == 0)
416 1.1 christos break;
417 1.1 christos
418 1.1 christos /* another iteration of the loop with odd and even swapped */
419 1.1 christos gf2_matrix_square(odd, even);
420 1.1 christos if (len2 & 1)
421 1.1 christos crc1 = gf2_matrix_times(odd, crc1);
422 1.1 christos len2 >>= 1;
423 1.1 christos
424 1.1 christos /* if no more bits set, then done */
425 1.1 christos } while (len2 != 0);
426 1.1 christos
427 1.1 christos /* return combined crc */
428 1.1 christos crc1 ^= crc2;
429 1.1 christos return crc1;
430 1.1 christos }
431