| // Copyright 2015 The Bazel Authors. All rights reserved. |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| package com.google.devtools.build.zip; |
| |
| import java.io.IOException; |
| import java.io.InputStream; |
| import java.util.zip.ZipException; |
| |
| class Zip64EndOfCentralDirectory { |
| static final int SIGNATURE = 0x06064b50; |
| static final int FIXED_DATA_SIZE = 56; |
| static final int SIGNATURE_OFFSET = 0; |
| static final int SIZE_OFFSET = 4; |
| static final int VERSION_OFFSET = 12; |
| static final int VERSION_NEEDED_OFFSET = 14; |
| static final int DISK_NUMBER_OFFSET = 16; |
| static final int CD_DISK_OFFSET = 20; |
| static final int DISK_ENTRIES_OFFSET = 24; |
| static final int TOTAL_ENTRIES_OFFSET = 32; |
| static final int CD_SIZE_OFFSET = 40; |
| static final int CD_OFFSET_OFFSET = 48; |
| |
| /** |
| * Read the Zip64 end of central directory record from the input stream and parse additional |
| * {@link ZipFileData} from it. |
| */ |
| static ZipFileData read(InputStream in, ZipFileData file) throws IOException { |
| if (file == null) { |
| throw new NullPointerException(); |
| } |
| |
| byte[] fixedSizeData = new byte[FIXED_DATA_SIZE]; |
| if (ZipUtil.readFully(in, fixedSizeData) != FIXED_DATA_SIZE) { |
| throw new ZipException( |
| "Unexpected end of file while reading Zip64 End of Central Directory Record."); |
| } |
| if (!ZipUtil.arrayStartsWith(fixedSizeData, ZipUtil.intToLittleEndian(SIGNATURE))) { |
| throw new ZipException(String.format( |
| "Malformed Zip64 End of Central Directory; does not start with %08x", SIGNATURE)); |
| } |
| file.setZip64(true); |
| file.setCentralDirectoryOffset(ZipUtil.getUnsignedLong(fixedSizeData, CD_OFFSET_OFFSET)); |
| file.setExpectedEntries(ZipUtil.getUnsignedLong(fixedSizeData, TOTAL_ENTRIES_OFFSET)); |
| return file; |
| } |
| |
| /** |
| * Generates the raw byte data of the Zip64 end of central directory record for the file. |
| */ |
| static byte[] create(ZipFileData file) { |
| byte[] buf = new byte[FIXED_DATA_SIZE]; |
| ZipUtil.intToLittleEndian(buf, SIGNATURE_OFFSET, SIGNATURE); |
| ZipUtil.longToLittleEndian(buf, SIZE_OFFSET, FIXED_DATA_SIZE - 12); |
| ZipUtil.shortToLittleEndian(buf, VERSION_OFFSET, (short) 0x2d); |
| ZipUtil.shortToLittleEndian(buf, VERSION_NEEDED_OFFSET, (short) 0x2d); |
| ZipUtil.intToLittleEndian(buf, DISK_NUMBER_OFFSET, 0); |
| ZipUtil.intToLittleEndian(buf, CD_DISK_OFFSET, 0); |
| ZipUtil.longToLittleEndian(buf, DISK_ENTRIES_OFFSET, file.getNumEntries()); |
| ZipUtil.longToLittleEndian(buf, TOTAL_ENTRIES_OFFSET, file.getNumEntries()); |
| ZipUtil.longToLittleEndian(buf, CD_SIZE_OFFSET, file.getCentralDirectorySize()); |
| ZipUtil.longToLittleEndian(buf, CD_OFFSET_OFFSET, file.getCentralDirectoryOffset()); |
| return buf; |
| } |
| } |