2007-03-22 10:30:00 -07:00
|
|
|
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
2012-05-21 04:12:37 -07:00
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
#ifndef nsZipArchive_h_
|
|
|
|
#define nsZipArchive_h_
|
|
|
|
|
2011-12-16 11:42:07 -08:00
|
|
|
#include "mozilla/Attributes.h"
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
#define ZIP_TABSIZE 256
|
2009-10-04 10:20:45 -07:00
|
|
|
#define ZIP_BUFLEN (4*1024) /* Used as output buffer when deflating items to a file */
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-09-08 20:37:11 -07:00
|
|
|
#ifndef PL_ARENA_CONST_ALIGN_MASK
|
2009-10-17 08:54:54 -07:00
|
|
|
#define PL_ARENA_CONST_ALIGN_MASK (sizeof(void*)-1)
|
2010-09-08 20:37:11 -07:00
|
|
|
#endif
|
2007-03-22 10:30:00 -07:00
|
|
|
#include "plarena.h"
|
|
|
|
|
|
|
|
#include "zlib.h"
|
2009-10-17 08:54:54 -07:00
|
|
|
#include "zipstruct.h"
|
2009-08-12 13:50:12 -07:00
|
|
|
#include "nsAutoPtr.h"
|
2012-06-05 19:08:30 -07:00
|
|
|
#include "nsIFile.h"
|
2010-08-18 10:34:07 -07:00
|
|
|
#include "mozilla/FileUtils.h"
|
2011-12-08 02:03:36 -08:00
|
|
|
#include "mozilla/FileLocation.h"
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-01-22 04:56:28 -08:00
|
|
|
#if defined(XP_WIN) && defined(_MSC_VER)
|
2010-11-11 12:13:48 -08:00
|
|
|
#define MOZ_WIN_MEM_TRY_BEGIN __try {
|
|
|
|
#define MOZ_WIN_MEM_TRY_CATCH(cmd) } \
|
|
|
|
__except(GetExceptionCode()==EXCEPTION_IN_PAGE_ERROR ? \
|
|
|
|
EXCEPTION_EXECUTE_HANDLER : EXCEPTION_CONTINUE_SEARCH) \
|
|
|
|
{ \
|
|
|
|
NS_WARNING("EXCEPTION_IN_PAGE_ERROR in " __FUNCTION__); \
|
|
|
|
cmd; \
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
#define MOZ_WIN_MEM_TRY_BEGIN {
|
|
|
|
#define MOZ_WIN_MEM_TRY_CATCH(cmd) }
|
|
|
|
#endif
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-11-11 12:13:48 -08:00
|
|
|
class nsZipFind;
|
2007-03-22 10:30:00 -07:00
|
|
|
struct PRFileDesc;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This file defines some of the basic structures used by libjar to
|
|
|
|
* read Zip files. It makes use of zlib in order to do the decompression.
|
|
|
|
*
|
|
|
|
* A few notes on the classes/structs:
|
|
|
|
* nsZipArchive represents a single Zip file, and maintains an index
|
|
|
|
* of all the items in the file.
|
|
|
|
* nsZipItem represents a single item (file) in the Zip archive.
|
|
|
|
* nsZipFind represents the metadata involved in doing a search,
|
|
|
|
* and current state of the iteration of found objects.
|
|
|
|
* 'MT''safe' reading from the zipfile is performed through JARInputStream,
|
|
|
|
* which maintains its own file descriptor, allowing for multiple reads
|
|
|
|
* concurrently from the same zip file.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* nsZipItem -- a helper struct for nsZipArchive
|
|
|
|
*
|
|
|
|
* each nsZipItem represents one file in the archive and all the
|
|
|
|
* information needed to manipulate it.
|
|
|
|
*/
|
2009-10-17 08:54:54 -07:00
|
|
|
class nsZipItem
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2009-10-17 08:54:54 -07:00
|
|
|
public:
|
|
|
|
const char* Name() { return ((const char*)central) + ZIPCENTRAL_SIZE; }
|
|
|
|
|
2009-12-15 15:01:08 -08:00
|
|
|
PRUint32 LocalOffset();
|
|
|
|
PRUint32 Size();
|
|
|
|
PRUint32 RealSize();
|
|
|
|
PRUint32 CRC32();
|
|
|
|
PRUint16 Date();
|
|
|
|
PRUint16 Time();
|
|
|
|
PRUint16 Compression();
|
|
|
|
bool IsDirectory();
|
|
|
|
PRUint16 Mode();
|
|
|
|
const PRUint8* GetExtraField(PRUint16 aTag, PRUint16 *aBlockSize);
|
|
|
|
PRTime LastModTime();
|
2007-03-22 10:30:00 -07:00
|
|
|
|
Bug 627277 - Remove (broken) BeOS support. r=biesi,dwitte,gavin,joe,jorendorff,josh,khuey,mfinkle,neil,Pike,roc,shaver,smontagu,taras
2011-02-19 11:10:24 -08:00
|
|
|
#ifdef XP_UNIX
|
2009-12-15 15:01:08 -08:00
|
|
|
bool IsSymlink();
|
2007-03-22 10:30:00 -07:00
|
|
|
#endif
|
|
|
|
|
2009-10-17 08:54:54 -07:00
|
|
|
nsZipItem* next;
|
|
|
|
const ZipCentral* central;
|
|
|
|
PRUint16 nameLength;
|
|
|
|
bool isSynthetic;
|
2007-03-22 10:30:00 -07:00
|
|
|
};
|
|
|
|
|
2009-08-12 13:50:12 -07:00
|
|
|
class nsZipHandle;
|
2009-10-04 10:20:45 -07:00
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
/**
|
|
|
|
* nsZipArchive -- a class for reading the PKZIP file format.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
class nsZipArchive
|
|
|
|
{
|
|
|
|
friend class nsZipFind;
|
|
|
|
|
|
|
|
public:
|
|
|
|
/** constructing does not open the archive. See OpenArchive() */
|
|
|
|
nsZipArchive();
|
|
|
|
|
|
|
|
/** destructing the object closes the archive */
|
|
|
|
~nsZipArchive();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* OpenArchive
|
|
|
|
*
|
|
|
|
* It's an error to call this more than once on the same nsZipArchive
|
|
|
|
* object. If we were allowed to use exceptions this would have been
|
|
|
|
* part of the constructor
|
|
|
|
*
|
2010-09-08 20:37:34 -07:00
|
|
|
* @param aZipHandle The nsZipHandle used to access the zip
|
2007-03-22 10:30:00 -07:00
|
|
|
* @return status code
|
|
|
|
*/
|
2010-09-08 20:37:34 -07:00
|
|
|
nsresult OpenArchive(nsZipHandle *aZipHandle);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* OpenArchive
|
|
|
|
*
|
|
|
|
* Convenience function that generates nsZipHandle
|
|
|
|
*
|
|
|
|
* @param aFile The file used to access the zip
|
|
|
|
* @return status code
|
|
|
|
*/
|
|
|
|
nsresult OpenArchive(nsIFile *aFile);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Test the integrity of items in this archive by running
|
|
|
|
* a CRC check after extracting each item into a memory
|
|
|
|
* buffer. If an entry name is supplied only the
|
|
|
|
* specified item is tested. Else, if null is supplied
|
|
|
|
* then all the items in the archive are tested.
|
|
|
|
*
|
|
|
|
* @return status code
|
|
|
|
*/
|
|
|
|
nsresult Test(const char *aEntryName);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Closes an open archive.
|
|
|
|
*/
|
|
|
|
nsresult CloseArchive();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* GetItem
|
|
|
|
* @param aEntryName Name of file in the archive
|
|
|
|
* @return pointer to nsZipItem
|
|
|
|
*/
|
|
|
|
nsZipItem* GetItem(const char * aEntryName);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* ExtractFile
|
|
|
|
*
|
|
|
|
* @param zipEntry Name of file in archive to extract
|
|
|
|
* @param outFD Filedescriptor to write contents to
|
|
|
|
* @param outname Name of file to write to
|
|
|
|
* @return status code
|
|
|
|
*/
|
|
|
|
nsresult ExtractFile(nsZipItem * zipEntry, const char *outname, PRFileDesc * outFD);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* FindInit
|
|
|
|
*
|
|
|
|
* Initializes a search for files in the archive. FindNext() returns
|
|
|
|
* the actual matches. The nsZipFind must be deleted when you're done
|
|
|
|
*
|
|
|
|
* @param aPattern a string or RegExp pattern to search for
|
|
|
|
* (may be NULL to find all files in archive)
|
|
|
|
* @param aFind a pointer to a pointer to a structure used
|
|
|
|
* in FindNext. In the case of an error this
|
|
|
|
* will be set to NULL.
|
|
|
|
* @return status code
|
|
|
|
*/
|
|
|
|
PRInt32 FindInit(const char * aPattern, nsZipFind** aFind);
|
|
|
|
|
2009-10-04 10:20:45 -07:00
|
|
|
/*
|
|
|
|
* Gets an undependent handle to the mapped file.
|
|
|
|
*/
|
|
|
|
nsZipHandle* GetFD();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get pointer to the data of the item.
|
|
|
|
* @param aItem Pointer to nsZipItem
|
|
|
|
* reutrns null when zip file is corrupt.
|
2007-03-22 10:30:00 -07:00
|
|
|
*/
|
2010-09-08 20:37:34 -07:00
|
|
|
const PRUint8* GetData(nsZipItem* aItem);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2012-02-22 10:45:09 -08:00
|
|
|
bool GetComment(nsACString &aComment);
|
|
|
|
|
2011-10-24 17:50:47 -07:00
|
|
|
/**
|
|
|
|
* Gets the amount of memory taken up by the archive's mapping.
|
|
|
|
* @return the size
|
|
|
|
*/
|
|
|
|
PRInt64 SizeOfMapping();
|
|
|
|
|
2011-12-08 02:03:36 -08:00
|
|
|
/*
|
|
|
|
* Refcounting
|
|
|
|
*/
|
|
|
|
NS_METHOD_(nsrefcnt) AddRef(void);
|
|
|
|
NS_METHOD_(nsrefcnt) Release(void);
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
private:
|
|
|
|
//--- private members ---
|
2011-12-08 02:03:36 -08:00
|
|
|
nsrefcnt mRefCnt; /* ref count */
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
nsZipItem* mFiles[ZIP_TABSIZE];
|
|
|
|
PLArenaPool mArena;
|
|
|
|
|
2012-02-22 10:45:09 -08:00
|
|
|
const char* mCommentPtr;
|
|
|
|
PRUint16 mCommentLen;
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
// Whether we synthesized the directory entries
|
2009-10-04 10:20:45 -07:00
|
|
|
bool mBuiltSynthetics;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2009-08-12 13:50:12 -07:00
|
|
|
// file handle
|
|
|
|
nsRefPtr<nsZipHandle> mFd;
|
2010-08-18 10:34:07 -07:00
|
|
|
|
|
|
|
// logging handle
|
|
|
|
mozilla::AutoFDClose mLog;
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-12-16 11:42:07 -08:00
|
|
|
private:
|
|
|
|
//--- private methods ---
|
2009-10-17 08:54:54 -07:00
|
|
|
nsZipItem* CreateZipItem();
|
2007-03-22 10:30:00 -07:00
|
|
|
nsresult BuildFileList();
|
|
|
|
nsresult BuildSynthetics();
|
2011-12-16 11:42:07 -08:00
|
|
|
|
|
|
|
nsZipArchive& operator=(const nsZipArchive& rhs) MOZ_DELETE;
|
|
|
|
nsZipArchive(const nsZipArchive& rhs) MOZ_DELETE;
|
2009-08-12 13:50:12 -07:00
|
|
|
};
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
/**
|
|
|
|
* nsZipFind
|
|
|
|
*
|
|
|
|
* a helper class for nsZipArchive, representing a search
|
|
|
|
*/
|
|
|
|
class nsZipFind
|
|
|
|
{
|
|
|
|
public:
|
2011-09-28 23:19:26 -07:00
|
|
|
nsZipFind(nsZipArchive* aZip, char* aPattern, bool regExp);
|
2007-03-22 10:30:00 -07:00
|
|
|
~nsZipFind();
|
|
|
|
|
2009-10-17 08:54:54 -07:00
|
|
|
nsresult FindNext(const char** aResult, PRUint16* aNameLen);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
private:
|
2011-12-08 02:03:36 -08:00
|
|
|
nsRefPtr<nsZipArchive> mArchive;
|
2007-03-22 10:30:00 -07:00
|
|
|
char* mPattern;
|
|
|
|
nsZipItem* mItem;
|
|
|
|
PRUint16 mSlot;
|
2011-09-28 23:19:26 -07:00
|
|
|
bool mRegExp;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-12-16 11:42:07 -08:00
|
|
|
nsZipFind& operator=(const nsZipFind& rhs) MOZ_DELETE;
|
|
|
|
nsZipFind(const nsZipFind& rhs) MOZ_DELETE;
|
2007-03-22 10:30:00 -07:00
|
|
|
};
|
|
|
|
|
2010-06-04 14:10:23 -07:00
|
|
|
/**
|
|
|
|
* nsZipCursor -- a low-level class for reading the individual items in a zip.
|
|
|
|
*/
|
|
|
|
class nsZipCursor {
|
|
|
|
public:
|
|
|
|
/**
|
|
|
|
* Initializes the cursor
|
|
|
|
*
|
|
|
|
* @param aItem Item of interest
|
|
|
|
* @param aZip Archive
|
|
|
|
* @param aBuf Buffer used for decompression.
|
|
|
|
* This determines the maximum Read() size in the compressed case.
|
|
|
|
* @param aBufSize Buffer size
|
|
|
|
* @param doCRC When set to true Read() will check crc
|
|
|
|
*/
|
|
|
|
nsZipCursor(nsZipItem *aItem, nsZipArchive *aZip, PRUint8* aBuf = NULL, PRUint32 aBufSize = 0, bool doCRC = false);
|
|
|
|
|
|
|
|
~nsZipCursor();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Performs reads. In the compressed case it uses aBuf(passed in constructor), for stored files
|
|
|
|
* it returns a zero-copy buffer.
|
|
|
|
*
|
|
|
|
* @param aBytesRead Outparam for number of bytes read.
|
|
|
|
* @return data read or NULL if item is corrupted.
|
|
|
|
*/
|
2011-12-08 02:03:36 -08:00
|
|
|
PRUint8* Read(PRUint32 *aBytesRead) {
|
|
|
|
return ReadOrCopy(aBytesRead, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Performs a copy. It always uses aBuf(passed in constructor).
|
|
|
|
*
|
|
|
|
* @param aBytesRead Outparam for number of bytes read.
|
|
|
|
* @return data read or NULL if item is corrupted.
|
|
|
|
*/
|
|
|
|
PRUint8* Copy(PRUint32 *aBytesRead) {
|
|
|
|
return ReadOrCopy(aBytesRead, true);
|
|
|
|
}
|
2010-06-04 14:10:23 -07:00
|
|
|
|
|
|
|
private:
|
2011-12-08 02:03:36 -08:00
|
|
|
/* Actual implementation for both Read and Copy above */
|
|
|
|
PRUint8* ReadOrCopy(PRUint32 *aBytesRead, bool aCopy);
|
|
|
|
|
2010-06-04 14:10:23 -07:00
|
|
|
nsZipItem *mItem;
|
|
|
|
PRUint8 *mBuf;
|
|
|
|
PRUint32 mBufSize;
|
|
|
|
z_stream mZs;
|
|
|
|
PRUint32 mCRC;
|
|
|
|
bool mDoCRC;
|
|
|
|
};
|
|
|
|
|
2010-09-08 20:37:34 -07:00
|
|
|
/**
|
2010-06-04 14:10:23 -07:00
|
|
|
* nsZipItemPtr - a RAII convenience class for reading the individual items in a zip.
|
|
|
|
* It reads whole files and does zero-copy IO for stored files. A buffer is allocated
|
|
|
|
* for decompression.
|
|
|
|
* Do not use when the file may be very large.
|
|
|
|
*/
|
|
|
|
class nsZipItemPtr_base {
|
|
|
|
public:
|
|
|
|
/**
|
|
|
|
* Initializes the reader
|
|
|
|
*
|
|
|
|
* @param aZip Archive
|
|
|
|
* @param aEntryName Archive membername
|
|
|
|
* @param doCRC When set to true Read() will check crc
|
|
|
|
*/
|
|
|
|
nsZipItemPtr_base(nsZipArchive *aZip, const char *aEntryName, bool doCRC);
|
|
|
|
|
|
|
|
PRUint32 Length() const {
|
|
|
|
return mReadlen;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
|
|
|
nsRefPtr<nsZipHandle> mZipHandle;
|
|
|
|
nsAutoArrayPtr<PRUint8> mAutoBuf;
|
|
|
|
PRUint8 *mReturnBuf;
|
|
|
|
PRUint32 mReadlen;
|
|
|
|
};
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
class nsZipItemPtr : public nsZipItemPtr_base {
|
|
|
|
public:
|
|
|
|
nsZipItemPtr(nsZipArchive *aZip, const char *aEntryName, bool doCRC = false) : nsZipItemPtr_base(aZip, aEntryName, doCRC) { }
|
|
|
|
/**
|
|
|
|
* @return buffer containing the whole zip member or NULL on error.
|
|
|
|
* The returned buffer is owned by nsZipItemReader.
|
|
|
|
*/
|
|
|
|
const T* Buffer() const {
|
|
|
|
return (const T*)mReturnBuf;
|
|
|
|
}
|
|
|
|
|
|
|
|
operator const T*() const {
|
|
|
|
return Buffer();
|
|
|
|
}
|
2010-11-11 12:13:57 -08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Relinquish ownership of zip member if compressed.
|
|
|
|
* Copy member into a new buffer if uncompressed.
|
|
|
|
* @return a buffer with whole zip member. It is caller's responsibility to free() it.
|
|
|
|
*/
|
|
|
|
T* Forget() {
|
|
|
|
if (!mReturnBuf)
|
|
|
|
return NULL;
|
|
|
|
// In uncompressed mmap case, give up buffer
|
|
|
|
if (mAutoBuf.get() == mReturnBuf) {
|
|
|
|
mReturnBuf = NULL;
|
|
|
|
return (T*) mAutoBuf.forget();
|
|
|
|
}
|
|
|
|
T *ret = (T*) malloc(Length());
|
|
|
|
memcpy(ret, mReturnBuf, Length());
|
|
|
|
mReturnBuf = NULL;
|
|
|
|
return ret;
|
|
|
|
}
|
2010-06-04 14:10:23 -07:00
|
|
|
};
|
|
|
|
|
2010-09-08 20:37:34 -07:00
|
|
|
class nsZipHandle {
|
|
|
|
friend class nsZipArchive;
|
2011-12-08 02:03:36 -08:00
|
|
|
friend class mozilla::FileLocation;
|
2010-09-08 20:37:34 -07:00
|
|
|
public:
|
2012-06-05 19:08:30 -07:00
|
|
|
static nsresult Init(nsIFile *file, nsZipHandle **ret NS_OUTPARAM);
|
2010-09-08 20:37:34 -07:00
|
|
|
static nsresult Init(nsZipArchive *zip, const char *entry,
|
|
|
|
nsZipHandle **ret NS_OUTPARAM);
|
|
|
|
|
|
|
|
NS_METHOD_(nsrefcnt) AddRef(void);
|
|
|
|
NS_METHOD_(nsrefcnt) Release(void);
|
|
|
|
|
2011-10-24 17:50:47 -07:00
|
|
|
PRInt64 SizeOfMapping();
|
|
|
|
|
2010-09-08 20:37:34 -07:00
|
|
|
protected:
|
|
|
|
const PRUint8 * mFileData; /* pointer to mmaped file */
|
|
|
|
PRUint32 mLen; /* length of file and memory mapped area */
|
2011-12-08 02:03:36 -08:00
|
|
|
mozilla::FileLocation mFile; /* source file if any, for logging */
|
2010-09-08 20:37:34 -07:00
|
|
|
|
|
|
|
private:
|
|
|
|
nsZipHandle();
|
|
|
|
~nsZipHandle();
|
|
|
|
|
|
|
|
PRFileMap * mMap; /* nspr datastructure for mmap */
|
|
|
|
nsAutoPtr<nsZipItemPtr<PRUint8> > mBuf;
|
|
|
|
nsrefcnt mRefCnt; /* ref count */
|
|
|
|
};
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
nsresult gZlibInit(z_stream *zs);
|
|
|
|
|
|
|
|
#endif /* nsZipArchive_h_ */
|