1 /*
2  * Licensed to the Apache Software Foundation (ASF) under one or more
3  * contributor license agreements.  See the NOTICE file distributed with
4  * this work for additional information regarding copyright ownership.
5  * The ASF licenses this file to You under the Apache License, Version 2.0
6  * (the "License"); you may not use this file except in compliance with
7  * the License.  You may obtain a copy of the License at
8  *
9  *     http://www.apache.org/licenses/LICENSE-2.0
10  *
11  * Unless required by applicable law or agreed to in writing, software
12  * distributed under the License is distributed on an "AS IS" BASIS,
13  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14  * See the License for the specific language governing permissions and
15  * limitations under the License.
16  */
17 /* Apache Harmony HEADER because the code in this class comes mostly from ZipFile, ZipEntry and
18  * ZipConstants from android libcore.
19  */
20 
21 package androidx.multidex;
22 
23 import java.io.File;
24 import java.io.IOException;
25 import java.io.RandomAccessFile;
26 import java.util.zip.CRC32;
27 import java.util.zip.ZipException;
28 
29 /**
30  * Tools to build a quick partial crc of zip files.
31  */
32 final class ZipUtil {
33     static class CentralDirectory {
34         long offset;
35         long size;
36     }
37 
38     /* redefine those constant here because of bug 13721174 preventing to compile using the
39      * constants defined in ZipFile */
40     private static final int ENDHDR = 22;
41     private static final int ENDSIG = 0x6054b50;
42 
43     /**
44      * Size of reading buffers.
45      */
46     private static final int BUFFER_SIZE = 0x4000;
47 
48     /**
49      * Compute crc32 of the central directory of an apk. The central directory contains
50      * the crc32 of each entries in the zip so the computed result is considered valid for the whole
51      * zip file. Does not support zip64 nor multidisk but it should be OK for now since ZipFile does
52      * not either.
53      */
getZipCrc(File apk)54     static long getZipCrc(File apk) throws IOException {
55         RandomAccessFile raf = new RandomAccessFile(apk, "r");
56         try {
57             CentralDirectory dir = findCentralDirectory(raf);
58 
59             return computeCrcOfCentralDir(raf, dir);
60         } finally {
61             raf.close();
62         }
63     }
64 
65     /* Package visible for testing */
findCentralDirectory(RandomAccessFile raf)66     static CentralDirectory findCentralDirectory(RandomAccessFile raf) throws IOException,
67             ZipException {
68         long scanOffset = raf.length() - ENDHDR;
69         if (scanOffset < 0) {
70             throw new ZipException("File too short to be a zip file: " + raf.length());
71         }
72 
73         long stopOffset = scanOffset - 0x10000 /* ".ZIP file comment"'s max length */;
74         if (stopOffset < 0) {
75             stopOffset = 0;
76         }
77 
78         int endSig = Integer.reverseBytes(ENDSIG);
79         while (true) {
80             raf.seek(scanOffset);
81             if (raf.readInt() == endSig) {
82                 break;
83             }
84 
85             scanOffset--;
86             if (scanOffset < stopOffset) {
87                 throw new ZipException("End Of Central Directory signature not found");
88             }
89         }
90         // Read the End Of Central Directory. ENDHDR includes the signature
91         // bytes,
92         // which we've already read.
93 
94         // Pull out the information we need.
95         raf.skipBytes(2); // diskNumber
96         raf.skipBytes(2); // diskWithCentralDir
97         raf.skipBytes(2); // numEntries
98         raf.skipBytes(2); // totalNumEntries
99         CentralDirectory dir = new CentralDirectory();
100         dir.size = Integer.reverseBytes(raf.readInt()) & 0xFFFFFFFFL;
101         dir.offset = Integer.reverseBytes(raf.readInt()) & 0xFFFFFFFFL;
102         return dir;
103     }
104 
105     /* Package visible for testing */
computeCrcOfCentralDir(RandomAccessFile raf, CentralDirectory dir)106     static long computeCrcOfCentralDir(RandomAccessFile raf, CentralDirectory dir)
107             throws IOException {
108         CRC32 crc = new CRC32();
109         long stillToRead = dir.size;
110         raf.seek(dir.offset);
111         int length = (int) Math.min(BUFFER_SIZE, stillToRead);
112         byte[] buffer = new byte[BUFFER_SIZE];
113         length = raf.read(buffer, 0, length);
114         while (length != -1) {
115             crc.update(buffer, 0, length);
116             stillToRead -= length;
117             if (stillToRead == 0) {
118                 break;
119             }
120             length = (int) Math.min(BUFFER_SIZE, stillToRead);
121             length = raf.read(buffer, 0, length);
122         }
123         return crc.getValue();
124     }
125 }
126