1 /* 2 * Copyright (C) 2006 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 // 18 // General-purpose Zip archive access. This class allows both reading and 19 // writing to Zip archives, including deletion of existing entries. 20 // 21 #ifndef __LIBS_ZIPFILE_H 22 #define __LIBS_ZIPFILE_H 23 24 #include <utils/Vector.h> 25 #include <utils/Errors.h> 26 #include <stdio.h> 27 28 #include "ZipEntry.h" 29 30 namespace android { 31 32 /* 33 * Manipulate a Zip archive. 34 * 35 * Some changes will not be visible in the until until "flush" is called. 36 * 37 * The correct way to update a file archive is to make all changes to a 38 * copy of the archive in a temporary file, and then unlink/rename over 39 * the original after everything completes. Because we're only interested 40 * in using this for packaging, we don't worry about such things. Crashing 41 * after making changes and before flush() completes could leave us with 42 * an unusable Zip archive. 43 */ 44 class ZipFile { 45 public: ZipFile(void)46 ZipFile(void) 47 : mZipFp(NULL), mReadOnly(false), mNeedCDRewrite(false) 48 {} ~ZipFile(void)49 ~ZipFile(void) { 50 if (!mReadOnly) 51 flush(); 52 if (mZipFp != NULL) 53 fclose(mZipFp); 54 discardEntries(); 55 } 56 57 /* 58 * Open a new or existing archive. 59 */ 60 enum { 61 kOpenReadOnly = 0x01, 62 kOpenReadWrite = 0x02, 63 kOpenCreate = 0x04, // create if it doesn't exist 64 kOpenTruncate = 0x08, // if it exists, empty it 65 }; 66 status_t open(const char* zipFileName, int flags); 67 68 /* 69 * Add a file to the end of the archive. Specify whether you want the 70 * library to try to store it compressed. 71 * 72 * If "storageName" is specified, the archive will use that instead 73 * of "fileName". 74 * 75 * If there is already an entry with the same name, the call fails. 76 * Existing entries with the same name must be removed first. 77 * 78 * If "ppEntry" is non-NULL, a pointer to the new entry will be returned. 79 */ add(const char * fileName,int compressionMethod,ZipEntry ** ppEntry)80 status_t add(const char* fileName, int compressionMethod, 81 ZipEntry** ppEntry) 82 { 83 return add(fileName, fileName, compressionMethod, ppEntry); 84 } add(const char * fileName,const char * storageName,int compressionMethod,ZipEntry ** ppEntry)85 status_t add(const char* fileName, const char* storageName, 86 int compressionMethod, ZipEntry** ppEntry) 87 { 88 return addCommon(fileName, NULL, 0, storageName, 89 ZipEntry::kCompressStored, 90 compressionMethod, ppEntry); 91 } 92 93 /* 94 * Add a file that is already compressed with gzip. 95 * 96 * If "ppEntry" is non-NULL, a pointer to the new entry will be returned. 97 */ addGzip(const char * fileName,const char * storageName,ZipEntry ** ppEntry)98 status_t addGzip(const char* fileName, const char* storageName, 99 ZipEntry** ppEntry) 100 { 101 return addCommon(fileName, NULL, 0, storageName, 102 ZipEntry::kCompressDeflated, 103 ZipEntry::kCompressDeflated, ppEntry); 104 } 105 106 /* 107 * Add a file from an in-memory data buffer. 108 * 109 * If "ppEntry" is non-NULL, a pointer to the new entry will be returned. 110 */ add(const void * data,size_t size,const char * storageName,int compressionMethod,ZipEntry ** ppEntry)111 status_t add(const void* data, size_t size, const char* storageName, 112 int compressionMethod, ZipEntry** ppEntry) 113 { 114 return addCommon(NULL, data, size, storageName, 115 ZipEntry::kCompressStored, 116 compressionMethod, ppEntry); 117 } 118 119 /* 120 * Add an entry by copying it from another zip file. If "padding" is 121 * nonzero, the specified number of bytes will be added to the "extra" 122 * field in the header. 123 * 124 * If "ppEntry" is non-NULL, a pointer to the new entry will be returned. 125 */ 126 status_t add(const ZipFile* pSourceZip, const ZipEntry* pSourceEntry, 127 int padding, ZipEntry** ppEntry); 128 129 /* 130 * Add an entry by copying it from another zip file, recompressing with 131 * Zopfli if already compressed. 132 * 133 * If "ppEntry" is non-NULL, a pointer to the new entry will be returned. 134 */ 135 status_t addRecompress(const ZipFile* pSourceZip, const ZipEntry* pSourceEntry, 136 ZipEntry** ppEntry); 137 138 /* 139 * Mark an entry as having been removed. It is not actually deleted 140 * from the archive or our internal data structures until flush() is 141 * called. 142 */ 143 status_t remove(ZipEntry* pEntry); 144 145 /* 146 * Flush changes. If mNeedCDRewrite is set, this writes the central dir. 147 */ 148 status_t flush(void); 149 150 /* 151 * Expand the data into the buffer provided. The buffer must hold 152 * at least <uncompressed len> bytes. Variation expands directly 153 * to a file. 154 * 155 * Returns "false" if an error was encountered in the compressed data. 156 */ 157 //bool uncompress(const ZipEntry* pEntry, void* buf) const; 158 //bool uncompress(const ZipEntry* pEntry, FILE* fp) const; 159 void* uncompress(const ZipEntry* pEntry) const; 160 161 /* 162 * Get an entry, by name. Returns NULL if not found. 163 * 164 * Does not return entries pending deletion. 165 */ 166 ZipEntry* getEntryByName(const char* fileName) const; 167 168 /* 169 * Get the Nth entry in the archive. 170 * 171 * This will return an entry that is pending deletion. 172 */ getNumEntries(void)173 int getNumEntries(void) const { return mEntries.size(); } 174 ZipEntry* getEntryByIndex(int idx) const; 175 176 private: 177 /* these are private and not defined */ 178 ZipFile(const ZipFile& src); 179 ZipFile& operator=(const ZipFile& src); 180 181 class EndOfCentralDir { 182 public: EndOfCentralDir(void)183 EndOfCentralDir(void) : 184 mDiskNumber(0), 185 mDiskWithCentralDir(0), 186 mNumEntries(0), 187 mTotalNumEntries(0), 188 mCentralDirSize(0), 189 mCentralDirOffset(0), 190 mCommentLen(0), 191 mComment(NULL) 192 {} ~EndOfCentralDir(void)193 virtual ~EndOfCentralDir(void) { 194 delete[] mComment; 195 } 196 197 status_t readBuf(const unsigned char* buf, int len); 198 status_t write(FILE* fp); 199 200 //unsigned long mSignature; 201 unsigned short mDiskNumber; 202 unsigned short mDiskWithCentralDir; 203 unsigned short mNumEntries; 204 unsigned short mTotalNumEntries; 205 unsigned long mCentralDirSize; 206 unsigned long mCentralDirOffset; // offset from first disk 207 unsigned short mCommentLen; 208 unsigned char* mComment; 209 210 enum { 211 kSignature = 0x06054b50, 212 kEOCDLen = 22, // EndOfCentralDir len, excl. comment 213 214 kMaxCommentLen = 65535, // longest possible in ushort 215 kMaxEOCDSearch = kMaxCommentLen + EndOfCentralDir::kEOCDLen, 216 217 }; 218 219 void dump(void) const; 220 }; 221 222 223 /* read all entries in the central dir */ 224 status_t readCentralDir(void); 225 226 /* crunch deleted entries out */ 227 status_t crunchArchive(void); 228 229 /* clean up mEntries */ 230 void discardEntries(void); 231 232 /* common handler for all "add" functions */ 233 status_t addCommon(const char* fileName, const void* data, size_t size, 234 const char* storageName, int sourceType, int compressionMethod, 235 ZipEntry** ppEntry); 236 237 /* copy all of "srcFp" into "dstFp" */ 238 status_t copyFpToFp(FILE* dstFp, FILE* srcFp, unsigned long* pCRC32); 239 /* copy all of "data" into "dstFp" */ 240 status_t copyDataToFp(FILE* dstFp, 241 const void* data, size_t size, unsigned long* pCRC32); 242 /* copy some of "srcFp" into "dstFp" */ 243 status_t copyPartialFpToFp(FILE* dstFp, FILE* srcFp, long length, 244 unsigned long* pCRC32); 245 /* like memmove(), but on parts of a single file */ 246 status_t filemove(FILE* fp, off_t dest, off_t src, size_t n); 247 /* compress all of "srcFp" into "dstFp", using Deflate */ 248 status_t compressFpToFp(FILE* dstFp, FILE* srcFp, 249 const void* data, size_t size, unsigned long* pCRC32); 250 251 /* get modification date from a file descriptor */ 252 time_t getModTime(int fd); 253 254 /* 255 * We use stdio FILE*, which gives us buffering but makes dealing 256 * with files >2GB awkward. Until we support Zip64, we're fine. 257 */ 258 FILE* mZipFp; // Zip file pointer 259 260 /* one of these per file */ 261 EndOfCentralDir mEOCD; 262 263 /* did we open this read-only? */ 264 bool mReadOnly; 265 266 /* set this when we trash the central dir */ 267 bool mNeedCDRewrite; 268 269 /* 270 * One ZipEntry per entry in the zip file. I'm using pointers instead 271 * of objects because it's easier than making operator= work for the 272 * classes and sub-classes. 273 */ 274 Vector<ZipEntry*> mEntries; 275 }; 276 277 }; // namespace android 278 279 #endif // __LIBS_ZIPFILE_H 280