emul/mini/src/main/java/java/util/zip/CRC32.java
author Jaroslav Tulach <jaroslav.tulach@apidesign.org>
Wed, 30 Jan 2013 14:03:49 +0100
branchemul
changeset 611 9839e9a75bcf
parent 609 48ef38e9677e
child 640 693745d01b55
permissions -rw-r--r--
Implementation of ZipInputStream
     1 /*
     2  * Copyright (c) 1996, 2005, Oracle and/or its affiliates. All rights reserved.
     3  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
     4  *
     5  * This code is free software; you can redistribute it and/or modify it
     6  * under the terms of the GNU General Public License version 2 only, as
     7  * published by the Free Software Foundation.  Oracle designates this
     8  * particular file as subject to the "Classpath" exception as provided
     9  * by Oracle in the LICENSE file that accompanied this code.
    10  *
    11  * This code is distributed in the hope that it will be useful, but WITHOUT
    12  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
    13  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
    14  * version 2 for more details (a copy is included in the LICENSE file that
    15  * accompanied this code).
    16  *
    17  * You should have received a copy of the GNU General Public License version
    18  * 2 along with this work; if not, write to the Free Software Foundation,
    19  * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
    20  *
    21  * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
    22  * or visit www.oracle.com if you need additional information or have any
    23  * questions.
    24  */
    25 
    26 package java.util.zip;
    27 
    28 /**
    29  * A class that can be used to compute the CRC-32 of a data stream.
    30  *
    31  * @see         Checksum
    32  * @author      David Connelly
    33  */
    34 public
    35 class CRC32 implements Checksum {
    36     private int crc = 0xFFFFFFFF;
    37 
    38     /**
    39      * Creates a new CRC32 object.
    40      */
    41     public CRC32() {
    42     }
    43 
    44 
    45     /**
    46      * Updates the CRC-32 checksum with the specified byte (the low
    47      * eight bits of the argument b).
    48      *
    49      * @param b the byte to update the checksum with
    50      */
    51     public void update(int b) {
    52         byte[] arr = { (byte)b };
    53         update(arr);
    54     }
    55 
    56     /**
    57      * Updates the CRC-32 checksum with the specified array of bytes.
    58      */
    59     public void update(byte[] b, int off, int len) {
    60         if (b == null) {
    61             throw new NullPointerException();
    62         }
    63         if (off < 0 || len < 0 || off > b.length - len) {
    64             throw new ArrayIndexOutOfBoundsException();
    65         }
    66         crc = updateBytes(crc, b, off, len);
    67     }
    68 
    69     /**
    70      * Updates the CRC-32 checksum with the specified array of bytes.
    71      *
    72      * @param b the array of bytes to update the checksum with
    73      */
    74     public void update(byte[] b) {
    75         crc = updateBytes(crc, b, 0, b.length);
    76     }
    77 
    78     /**
    79      * Resets CRC-32 to initial value.
    80      */
    81     public void reset() {
    82         crc = 0;
    83     }
    84 
    85     /**
    86      * Returns CRC-32 value.
    87      */
    88     public long getValue() {
    89         return (long)crc & 0xffffffffL;
    90     }
    91 
    92     // XXX: taken from 
    93     // http://introcs.cs.princeton.edu/java/51data/CRC32.java.html
    94     private static int updateBytes(int crc, byte[] arr, int off, int len) {
    95         int poly = 0xEDB88320;   // reverse polynomial
    96 
    97         while (len-- > 0) {
    98             byte b = arr[off++];
    99             int temp = (crc ^ b) & 0xff;
   100 
   101             // read 8 bits one at a time
   102             for (int i = 0; i < 8; i++) {
   103                 if ((temp & 1) == 1) {
   104                     temp = (temp >>> 1) ^ poly;
   105                 } else {
   106                     temp = (temp >>> 1);
   107                 }
   108             }
   109             crc = (crc >>> 8) ^ temp;
   110         }
   111         return crc ^ 0xffffffff;
   112     }
   113 }