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