First submission to Symbian Foundation staging server.
// fzipup.cpp
//
// Copyright (c) 2008 - 2010 Accenture. All rights reserved.
// This component and the accompanying materials are made available
// under the terms of the "Eclipse Public License v1.0"
// which accompanies this distribution, and is available
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
//
// Initial Contributors:
// Accenture - Initial contribution
//
#include <ezstream.h>
#include <ezcompressor.h>
#include <ezdecompressor.h>
#include <ezfilebuffer.h>
#include <f32file.h>
#include "fzipup.h"
const TInt KDefaultDiskNumber = 0;
const TInt CZipEntry::iOffset = _FOFF(CZipEntry, iLink);
const TUint16 KFZipVersion = 0x0014;
TInt KWindowBits = -CEZCompressor::EMaxWBits; // negative value is an undocumented feature of our zlib implementation, ensuring we do not include the zlib header in the compressed data (which we must do for zip-file format support)
const TInt KCompressedSizeRelativePosition = 18; // iCompressedSize is 18 bytes in from the current entry's LFH
//
// CBufferManager
//
CBufferManager *CBufferManager::NewLC(RFileReadStream& aInput, RFileWriteStream& aOutput, TInt aInputStreamBytes, TInt aBufferSize)
{
CBufferManager *bm = new (ELeave) CBufferManager(aInput, aOutput, aInputStreamBytes, aBufferSize);
CleanupStack::PushL(bm);
bm->ConstructL();
return bm;
}
CBufferManager::CBufferManager(RFileReadStream& aInput, RFileWriteStream& aOutput, TInt aInputStreamBytes, TInt aBufferSize)
:iInput(aInput), iOutput(aOutput), iInputStreamBytes(aInputStreamBytes), iBufferSize(aBufferSize), iInputDescriptor(NULL,0), iOutputDescriptor(NULL,0)
{
}
CBufferManager::~CBufferManager()
{
delete[] iInputBuffer;
delete[] iOutputBuffer;
}
void CBufferManager::ConstructL()
{
if (iInputStreamBytes < iBufferSize)
iBufferSize = iInputStreamBytes;
if (iBufferSize <=0)
{
User::Leave(KErrUnderflow);
}
iInputBuffer = new (ELeave) TUint8[iBufferSize];
iOutputBuffer = new (ELeave) TUint8[iBufferSize];
iInputDescriptor.Set(iInputBuffer,0,iBufferSize);
iOutputDescriptor.Set(iOutputBuffer,0,iBufferSize);
iReadLength = iBufferSize;
}
void CBufferManager::InitializeL(CEZZStream &aZStream)
{
ReadInputL();
aZStream.SetInput(iInputDescriptor);
aZStream.SetOutput(iOutputDescriptor);
}
void CBufferManager::NeedInputL(CEZZStream &aZStream)
{
ReadInputL();
aZStream.SetInput(iInputDescriptor);
}
void CBufferManager::NeedOutputL(CEZZStream &aZStream)
{
WriteOutputL(aZStream);
aZStream.SetOutput(iOutputDescriptor);
}
void CBufferManager::FinalizeL(CEZZStream &aZStream)
{
WriteOutputL(aZStream);
}
void CBufferManager::ReadInputL()
{
iInput.ReadL(iInputDescriptor, iReadLength);
iInputStreamBytes -= iReadLength;
if (iInputStreamBytes < iBufferSize)
{
iReadLength = iInputStreamBytes;
}
if (iReadLength < 0)
{
User::Leave(KErrOverflow);
}
}
void CBufferManager::WriteOutputL(CEZZStream& aZStream)
{
TPtrC8 od = aZStream.OutputDescriptor();
iOutput.WriteL(od);
iBytesWritten += od.Size();
}
//
// CZipEntry
//
CZipEntry* CZipEntry::NewLC(RFs& aFs, const TDesC& aFilename, const TInt aUid)
{
CZipEntry* self = new (ELeave) CZipEntry(aFs, aUid);
CleanupStack::PushL(self);
self->ConstructL(aFilename);
return self;
}
CZipEntry::CZipEntry(RFs& aFs, const TInt aUid):
iFs(aFs), iUid(aUid)
{
iLFH.iCRC32 = crc32(iLFH.iCRC32, NULL, 0);
}
CZipEntry::~CZipEntry()
{
if (iFileData)
delete iFileData;
if (iInput.SubSessionHandle() > 0)
iInput.Close();
}
void CZipEntry::ConstructL(const TDesC& aFilename)
{
User::LeaveIfError(iInput.Open(iFs, aFilename, EFileRead | EFileShareReadersOnly));
// ensure aFileName is stored in accordance with zip file-format specification
TFileName2 file(aFilename);
if (file.HasDriveLetter())
{
file.Delete(0, 3); // remove '<drive>:\'
}
if (file.HasLeadingSlash())
{
file.Delete(0, 1); // remove '\'
}
iAsciiName.Copy(file);
TUint8* ptr = const_cast<TUint8*> (iAsciiName.Ptr());
TInt count = 0;
do
{
if (*ptr == '\\')
{
*ptr = '/';
}
ptr++;
} while (count++ < iAsciiName.Length());
// config. local file header
iLFH.iSignature = KLocalHeaderSignature;
iLFH.iVersionNeeded = KFZipVersion;
iLFH.iFlags = 0x0002; // best (-exx/-ex) compression was used
iLFH.iCompressionMethod = EDeflated;
TTime time;
User::LeaveIfError(iInput.Modified(time));
TDateTime td = time.DateTime();
TUint16 param1 = td.Year() - 1980;
TUint16 param2 = td.Month() + 1;
TUint16 param3 = td.Day() + 1;
iLFH.iLastModifiedFileDate = (param1 << 9) | (param2 << 5) | param3;
param1 = td.Hour() + 1;
param2 = td.Minute() + 1;
param3 = td.Second() + 1;
iLFH.iLastModifiedFileTime = (param1 << 11) | (param2 << 5) | (param3 >> 1);
TInt uSize = 0;
User::LeaveIfError(iInput.Size(uSize));
if (uSize <= 0)
{
User::Leave(KErrAbort);
}
iLFH.iUncompressedSize = uSize;
iLFH.iCompressedSize = uSize; // note: to be adjusted later once the compression is performed
CalcCRCL();
iLFH.iFileNameLength = iAsciiName.Size();
iLFH.iExtraFieldLength = 0;
// config. file header
iFH.iSignature = KCentralDirectoryHeaderSignature;
iFH.iMadeBy = KFZipVersion;
iFH.iRequired = KFZipVersion;
iFH.iFlags = iLFH.iFlags;
iFH.iCompressionMethod = iLFH.iCompressionMethod;
iFH.iLastModifiedFileTime = iLFH.iLastModifiedFileTime;
iFH.iLastModifiedFileDate = iLFH.iLastModifiedFileDate;
iFH.iCRC32 = iLFH.iCRC32;
iFH.iCompressedSize = iLFH.iCompressedSize; // note: to be adjusted later once the compression is performed
iFH.iUncompressedSize = iLFH.iUncompressedSize;
iFH.iFileNameLength = iLFH.iFileNameLength;
iFH.iExtraFieldLength = iLFH.iExtraFieldLength;
iFH.iFileCommentLength = 0x0000;
iFH.iDiskNumberStart = KDefaultDiskNumber;
iFH.iInternalFileAttributes = 0x0001;
iFH.iExternalFileAttributes = 0x00000020; // hardcoded to 'Archive'
iFH.iLocalHeaderOffset = 0x00000000; // to be adjusted later (if necessary)
}
void CZipEntry::SetOffset(TUint32 aOffset)
{
iFH.iLocalHeaderOffset = aOffset;
}
TUint32 CZipEntry::ReturnOffset()
{
TUint32 result = iFH.iLocalHeaderOffset; // start with the offset of the LFH
result += KLocalHeaderFixedLength; // add the size of the LFH itself
result += iAsciiName.Size(); // add the size of the filename
result += iLFH.iCompressedSize; // add any compressed data
return result;
}
TUint32 CZipEntry::FileHeaderSize()
{
TUint32 result = KCentralDirectoryHeaderFixedLength;
result += iAsciiName.Size();
return result;
}
//
// CZipEntry::CalcCRCL
// mem. efficient means of calculating the crc on uncompressed data
// we require a maximum of KDefaultZipBufferLength heap space regardless size of data
//
void CZipEntry::CalcCRCL()
{
ASSERT(iInput.SubSessionHandle() > 0);
TInt bytesRead = 0;
TInt length = KDefaultZipBufferLength; // 32Kb
if (iLFH.iUncompressedSize < KDefaultZipBufferLength)
{
length = iLFH.iUncompressedSize;
}
iLFH.iCRC32 = crc32(0, NULL, 0);
HBufC8* crc = HBufC8::NewLC(length);
TPtr8 crcPtr = crc->Des();
User::LeaveIfError(iInput.Seek(ESeekStart, bytesRead));
do
{
User::LeaveIfError(iInput.Read(crcPtr, length));
iLFH.iCRC32 = crc32(iLFH.iCRC32, crcPtr.Ptr(), length);
bytesRead += length;
if ((iLFH.iUncompressedSize - bytesRead) < KDefaultZipBufferLength)
{
length = iLFH.iUncompressedSize - bytesRead;
}
} while (length > 0);
CleanupStack::PopAndDestroy(1); // crc
}
//
// CZipItUp
//
CZipItUp* CZipItUp::NewLC(RFs& aFs, TDesC& aArchive)
{
CZipItUp* self = new (ELeave) CZipItUp(aFs, aArchive);
CleanupStack::PushL(self);
self->ConstructL();
return self;
}
CZipItUp::CZipItUp(RFs& aFs, TDesC& aArchive):
iFs(aFs), iFileName(aArchive), iZipMemberLinkedList(CZipEntry::iOffset), iZipMemberLinkedListIter(iZipMemberLinkedList)
{
}
CZipItUp::~CZipItUp()
{
if (iFile.SubSessionHandle() > 0)
{
iFile.Close();
}
while (!iZipMemberLinkedList.IsEmpty())
{
CZipEntry* entry = iZipMemberLinkedList.First();
iZipMemberLinkedList.Remove(*entry);
delete entry;
}
iZipMemberLinkedList.Reset();
}
void CZipItUp::ConstructL()
{
User::LeaveIfError(iFile.Create(iFs, iFileName, EFileStream | EFileShareExclusive));
}
//
// CZipItUp::AddFileL
//
void CZipItUp::AddFileL(const TDesC& aFile)
{
if (!DuplicateEntryL(aFile))
{
// spawn a new zip entry, compress the file, add it to the list of zip file contained in the archive
CZipEntry* zipEntry = CZipEntry::NewLC(iFs, aFile, ++iEntryUid);
AddEntryL(*zipEntry);
CleanupStack::Pop(zipEntry);
}
}
//
// CZipItUp::DuplicateEntryL
// run a check against the zip archive's current entries (if any!) for duplicate files
// return ETrue if a duplicate is found
//
TBool CZipItUp::DuplicateEntryL(const TDesC& aFile)
{
if (iZipMemberLinkedList.IsEmpty())
{
return EFalse;
}
// turn aFile into a zip-file format filename
TFileName2 file(aFile);
if (file.HasDriveLetter())
{
file.Delete(0, 3); // remove '<drive>:\'
}
if (file.HasLeadingSlash())
{
file.Delete(0, 1); // remove '\'
}
TBuf8<KMaxFileName> ascii;
ascii.Copy(file);
TUint8* ptr = const_cast<TUint8*> (ascii.Ptr());
TInt count = 0;
do
{
if (*ptr == '\\')
{
*ptr = '/';
}
ptr++;
} while (count++ < ascii.Length());
// iterate through zip archive entries looking for a duplicate
iZipMemberLinkedListIter.SetToFirst();
CZipEntry* entry = iZipMemberLinkedListIter++;
while (entry != NULL)
{
if (entry->iAsciiName == ascii)
{
return ETrue;
}
entry = iZipMemberLinkedListIter++;
}
return EFalse;
}
//
// CZipItUp::AddEntryL
// Add an entry to the zip archive.
//
void CZipItUp::AddEntryL(CZipEntry& aEntry)
{
// append the new entry to the queue
if (!iZipMemberLinkedList.IsEmpty())
{
// prior entries in the list therefore need to adjust this entry's local file header offset
CZipEntry* lastEntry = iZipMemberLinkedList.Last();
User::LeaveIfNull(lastEntry);
}
iZipMemberLinkedList.AddLast(aEntry);
}
//
// CZipItUp::CreateZipL
// Creates the actual file, including all the previously compressed zip files in the archive
//
void CZipItUp::CreateZipL()
{
ASSERT(iFile.SubSessionHandle() > 0);
if (iZipMemberLinkedList.IsEmpty())
{
User::Leave(KErrGeneral);
}
RFileWriteStream stream(iFile);
stream.PushL();
// initialise central directory header
iCDT.iSignature = CZipArchive::KCentralDirectorySignature;
iCDT.iDiskNumber = KDefaultDiskNumber;
iCDT.iStartDiskNumber = KDefaultDiskNumber;
iCDT.iLocalEntryCount = 0;
iCDT.iTotalEntryCount = 0;
iCDT.iSize = 0;
iCDT.iCommentLength = 0;
// iterate through each zip entry, appending local file header (LFH), file data (FD) to file
iZipMemberLinkedListIter.SetToFirst();
CZipEntry* entry = iZipMemberLinkedListIter++;
CZipEntry* prior = NULL;
User::LeaveIfNull(entry);
do
{
if (prior)
{
// adjust file header local offset if necessary
entry->SetOffset(prior->ReturnOffset());
}
// insert entry's Local File Header
AppendLocalFileHeaderL(stream, *entry);
// insert entry's File Data
AppendCompressedDataL(stream, *entry);
// adjust cdt parameters
iCDT.iLocalEntryCount++;
iCDT.iTotalEntryCount++;
iCDT.iSize += entry->FileHeaderSize();
// move onto the next entry
prior = entry;
entry = iZipMemberLinkedListIter++;
} while (entry != NULL);
// adjust cdt offset parameter
iCDT.iOffset = iZipMemberLinkedList.Last()->ReturnOffset();
// iterate through each zip entry, appending the central directory File Headers to file (these must go after LFH, FD entries)
iZipMemberLinkedListIter.SetToFirst();
entry = iZipMemberLinkedListIter++;
do
{
// central directory File Header
AppendCentralDirectoryFileHeaderL(stream, *entry);
// move onto the next entry
entry = iZipMemberLinkedListIter++;
} while (entry != NULL);
// append the central directory trailer
AppendCentralDirectoryTrailerL(stream);
// commit the transaction
stream.CommitL();
stream.Pop();
stream.Close();
// go back through & adjust compressed size values for each zip entry post-mortem
ASSERT(iFile.SubSessionHandle() == 0);
User::LeaveIfError(iFile.Open(iFs, iFileName, EFileWrite | EFileShareExclusive));
TInt pos = 0;
const TInt remainder = 8; // CZipArchive::KLocalHeaderFixedLength - KCompressedSizeRelativePosition
User::LeaveIfError(iFile.Seek(ESeekStart, pos));
iZipMemberLinkedListIter.SetToFirst();
entry = iZipMemberLinkedListIter++;
do
{
pos += KCompressedSizeRelativePosition; // step to iCompressedSize location in the current entry's LFH
TUint32 cs = entry->iLFH.iCompressedSize;
TUint32 KMyByte = 0x000000FF;
TInt bytecount = 0;
TBuf8<8> compSize;
do
{
TUint32 csChar((cs & KMyByte));
compSize.Append(csChar);
cs >>= 8;
} while (++bytecount < 4); // zip-file format allows us 4 bytes only for the compressed size value
User::LeaveIfError(iFile.Write(pos, compSize));
pos += bytecount; // the 4 bytes we just wrote
// step over remaing bytes to the beginning of the next entry's LFH
pos += remainder;
pos += entry->iAsciiName.Size();
pos += entry->iLFH.iCompressedSize;
// we're now positioned at the next entry's LFH
entry = iZipMemberLinkedListIter++;
} while (entry != NULL);
}
void CZipItUp::AppendLocalFileHeaderL(RFileWriteStream& aStream, CZipEntry& aZipEntry)
{
aStream.WriteUint32L(aZipEntry.iLFH.iSignature);
aStream.WriteUint16L(aZipEntry.iLFH.iVersionNeeded);
aStream.WriteUint16L(aZipEntry.iLFH.iFlags);
aStream.WriteUint16L(aZipEntry.iLFH.iCompressionMethod);
aStream.WriteUint16L(aZipEntry.iLFH.iLastModifiedFileTime);
aStream.WriteUint16L(aZipEntry.iLFH.iLastModifiedFileDate);
aStream.WriteUint32L(aZipEntry.iLFH.iCRC32);
aStream.WriteUint32L(aZipEntry.iLFH.iCompressedSize);
aStream.WriteUint32L(aZipEntry.iLFH.iUncompressedSize);
aStream.WriteUint16L(aZipEntry.iLFH.iFileNameLength);
aStream.WriteUint16L(aZipEntry.iLFH.iExtraFieldLength);
aStream.WriteL(aZipEntry.iAsciiName);
}
void CZipItUp::AppendCompressedDataL(RFileWriteStream& aStream, CZipEntry& aZipEntry)
{
// compress data & stream it to the zip archive (aStream)
RFileReadStream inStream;
inStream.Attach(aZipEntry.iInput); // note takes ownership of iInput subsession handle
CBufferManager* fb = CBufferManager::NewLC(inStream, aStream, aZipEntry.iLFH.iUncompressedSize, KDefaultZipBufferLength);
CEZCompressor* compressor = CEZCompressor::NewLC(*fb, CEZCompressor::EBestCompression, KWindowBits, CEZCompressor::EDefMemLevel, CEZCompressor::EDefaultStrategy);
while (compressor->DeflateL()){/* do nothing */}
// update the entry's LocalFileHeader and FileHeader
aZipEntry.iLFH.iCompressedSize = fb->TotalBytesOut();
aZipEntry.iFH.iCompressedSize = aZipEntry.iLFH.iCompressedSize;
// cleanup
CleanupStack::PopAndDestroy(2, fb); // compressor, fb
inStream.Close();
}
void CZipItUp::AppendCentralDirectoryFileHeaderL(RFileWriteStream& aStream, CZipEntry& aEntry)
{
aStream.WriteUint32L(aEntry.iFH.iSignature);
aStream.WriteUint16L(aEntry.iFH.iMadeBy);
aStream.WriteUint16L(aEntry.iFH.iRequired);
aStream.WriteUint16L(aEntry.iFH.iFlags);
aStream.WriteUint16L(aEntry.iFH.iCompressionMethod);
aStream.WriteUint16L(aEntry.iFH.iLastModifiedFileTime);
aStream.WriteUint16L(aEntry.iFH.iLastModifiedFileDate);
aStream.WriteUint32L(aEntry.iFH.iCRC32);
aStream.WriteUint32L(aEntry.iFH.iCompressedSize);
aStream.WriteUint32L(aEntry.iFH.iUncompressedSize);
aStream.WriteUint16L(aEntry.iFH.iFileNameLength);
aStream.WriteUint16L(aEntry.iFH.iExtraFieldLength);
aStream.WriteUint16L(aEntry.iFH.iFileCommentLength);
aStream.WriteUint16L(aEntry.iFH.iDiskNumberStart);
aStream.WriteUint16L(aEntry.iFH.iInternalFileAttributes);
aStream.WriteUint32L(aEntry.iFH.iExternalFileAttributes);
aStream.WriteUint32L(aEntry.iFH.iLocalHeaderOffset);
aStream.WriteL(aEntry.iAsciiName);
}
void CZipItUp::AppendCentralDirectoryTrailerL(RFileWriteStream& aStream)
{
aStream.WriteUint32L(iCDT.iSignature);
aStream.WriteUint16L(iCDT.iDiskNumber);
aStream.WriteUint16L(iCDT.iStartDiskNumber);
aStream.WriteUint16L(iCDT.iLocalEntryCount);
aStream.WriteUint16L(iCDT.iTotalEntryCount);
aStream.WriteUint32L(iCDT.iSize);
aStream.WriteUint32L(iCDT.iOffset);
aStream.WriteUint16L(iCDT.iCommentLength);
}