Update contrib.
1 // Copyright (c) 2003-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
19 const TUint8 EZGZipFile::ID1 = 31;
20 const TUint8 EZGZipFile::ID2 = 139;
25 EXPORT_C TEZGZipHeader::TEZGZipHeader() : iId1(EZGZipFile::ID1), iId2(EZGZipFile::ID2), iCompressionMethod(8), iFlags(0), iTime(0),
26 iExtraFlags(0), iOs(255), iXlen(0), iExtra(NULL), iFname(NULL), iComment(NULL), iCrc(0)
34 EXPORT_C TEZGZipHeader::~TEZGZipHeader()
44 EXPORT_C TEZGZipTrailer::TEZGZipTrailer() : iCrc32(0), iSize(0)
52 @param aCrc the CRC to use for archive checking
53 @param aSize the size of the trailer
55 EXPORT_C TEZGZipTrailer::TEZGZipTrailer(TInt32 aCrc, TInt32 aSize) : iCrc32(aCrc), iSize(aSize)
60 //--------------------------------------------------------------------------------------------------------
63 Read the zip header from the specified zip file into the TEZGZipHeader object
65 @param aFile the zip file to read from
66 @param aHeader the target header object
67 @leave KEZlibErrBadGZipHeader invalid zip header
68 @leave ... Any of the system wide error codes
70 EXPORT_C void EZGZipFile::ReadHeaderL(RFile &aFile, TEZGZipHeader &aHeader)
72 TInt obligatoryData = sizeof(aHeader.iId1) + sizeof(aHeader.iId2) + sizeof(aHeader.iCompressionMethod) +
73 sizeof(aHeader.iFlags) + sizeof(aHeader.iTime) + sizeof(aHeader.iExtraFlags) + sizeof(aHeader.iOs);
75 TPtr8 des(&aHeader.iId1,0,obligatoryData);
76 TInt err = aFile.Read(des);
77 if (err != KErrNone || (des.Size() != obligatoryData))
78 User::Leave(KEZlibErrBadGZipHeader);
80 if (aHeader.iId1 != ID1 || aHeader.iId2 != ID2)
81 User::Leave(KEZlibErrBadGZipHeader);
83 if (aHeader.iFlags & (1 << EFExtra)) // then the extra bit is set
85 des.Set(REINTERPRET_CAST(TUint8 *,&aHeader.iXlen),0,sizeof(aHeader.iXlen));
86 err = aFile.Read(des);
87 if (err != KErrNone || des.Size() != sizeof(aHeader.iXlen) || aHeader.iXlen < 0)
88 User::Leave(KEZlibErrBadGZipHeader);
90 aHeader.iExtra = HBufC8::NewMaxL(aHeader.iXlen);
91 TPtr8 des = aHeader.iExtra->Des();
92 err = aFile.Read(des);
93 if (err != KErrNone || des.Size() != aHeader.iXlen)
94 User::Leave(KEZlibErrBadGZipHeader);
97 if (aHeader.iFlags & (1 << EFName)) // then read in filename
98 ReadStringIntoDescriptorL(aFile,&aHeader.iFname);
100 if (aHeader.iFlags & (1 << EFComment)) // then read in comment
101 ReadStringIntoDescriptorL(aFile,&aHeader.iComment);
103 if (aHeader.iFlags & (1 << EFHcrc))
105 des.Set(REINTERPRET_CAST(TUint8*,&aHeader.iCrc),0,sizeof(aHeader.iCrc));
106 err = aFile.Read(des);
107 if (err != KErrNone || des.Size() != sizeof(aHeader.iCrc))
108 User::Leave(KEZlibErrBadGZipHeader);
113 Write the zip header to the specified file
115 @param aFile the file to write to
116 @param aHeader the header object to write to the file
117 @leave ... Any of the system wide error codes
119 EXPORT_C void EZGZipFile::WriteHeaderL(RFile &aFile, TEZGZipHeader &aHeader)
121 TInt obligatoryData = sizeof(aHeader.iId1) + sizeof(aHeader.iId2) + sizeof(aHeader.iCompressionMethod) +
122 sizeof(aHeader.iFlags) + sizeof(aHeader.iTime) + sizeof(aHeader.iExtraFlags) + sizeof(aHeader.iOs);
124 TPtrC8 des(&aHeader.iId1,obligatoryData);
125 User::LeaveIfError(aFile.Write(des));
129 if (aHeader.iFlags & (1 << EFExtra)) // then the extra bit is set
131 des.Set(REINTERPRET_CAST(TUint8 *,&aHeader.iXlen),sizeof(aHeader.iXlen));
132 User::LeaveIfError(aFile.Write(des));
134 User::LeaveIfError(aFile.Write(*aHeader.iExtra));
137 if (aHeader.iFlags & (1 << EFName)) // then read in filename
139 User::LeaveIfError(aFile.Write(*aHeader.iFname));
140 User::LeaveIfError(aFile.Write(null));
143 if (aHeader.iFlags & (1 << EFComment)) // then read in comment
145 User::LeaveIfError(aFile.Write(*aHeader.iComment));
146 User::LeaveIfError(aFile.Write(null));
149 if (aHeader.iFlags & (1 << EFHcrc))
151 des.Set(REINTERPRET_CAST(TUint8*,&aHeader.iCrc),sizeof(aHeader.iCrc));
152 User::LeaveIfError(aFile.Write(des));
156 void EZGZipFile::ReadStringIntoDescriptorL(RFile &aFile, HBufC8 **aDes)
159 CArrayFixFlat<TUint8> *bytes = new (ELeave) CArrayFixFlat<TUint8>(16);
160 CleanupStack::PushL(bytes);
162 while ((err = aFile.Read(ch)) == KErrNone)
164 if (ch.Length() != 1)
165 User::Leave(KEZlibErrBadGZipHeader);
166 else if (ch[0] != '\0')
167 bytes->AppendL(*ch.Ptr());
173 User::Leave(KEZlibErrBadGZipHeader);
175 *aDes = HBufC8::NewMaxL(bytes->Count());
177 for (i = 0; i < bytes->Count(); i++)
178 (*aDes)->Des()[i] = (*bytes)[i];
180 CleanupStack::PopAndDestroy(); // delete bytes
184 Read the zip trailer from the specified zip file into the TEZGZipTrailer object
186 @param aFile the zip file to read from
187 @param aTrailer the target trailer object
188 @leave KEZlibErrBadGZipTrailer invalid zip trailer
189 @leave ... Any of the system wide error codes
191 EXPORT_C void EZGZipFile::ReadTrailerL(RFile &aFile, TEZGZipTrailer &aTrailer)
193 TPtr8 des(REINTERPRET_CAST(TUint8*,&aTrailer.iCrc32),0,sizeof(TEZGZipTrailer));
195 TInt err = aFile.Read(des);
196 if (err != KErrNone || des.Size() != sizeof(TEZGZipTrailer))
197 User::Leave(KEZlibErrBadGZipTrailer);
201 Write the zip trailer to the specified file
203 @param aFile the file to write to
204 @param aTrailer the trailer object to write to the file
205 @leave ... Any of the system wide error codes
207 EXPORT_C void EZGZipFile::WriteTrailerL(RFile &aFile, TEZGZipTrailer &aTrailer)
209 TPtrC8 des(REINTERPRET_CAST(TUint8*,&aTrailer.iCrc32),sizeof(TEZGZipTrailer));
210 User::LeaveIfError(aFile.Write(des));
214 Find the zip trailer within the specified file, and read it into the TEZGZipTrailer object
216 @param aRfs file server session
217 @param aFname the file to read from
218 @param aTrailer the target trailer object
219 @leave KEZlibErrBadGZipHeader Invalid zip header
220 @leave ... Any of the system wide error codes
222 EXPORT_C void EZGZipFile::LocateAndReadTrailerL(RFs &aRfs, const TDesC &aFname, TEZGZipTrailer &aTrailer)
228 User::LeaveIfError(file.Open(aRfs,aFname,EFileStream | EFileRead | EFileShareAny));
229 CleanupClosePushL(file);
232 TPtr8 des(magic,0,sizeof(TUint8) * 2);
233 User::LeaveIfError(file.Read(des));
234 if (magic[0] != ID1 || magic[1] != ID2)
235 User::Leave(KEZlibErrBadGZipHeader);
237 User::LeaveIfError(file.Size(fileSize));
238 TInt sizePos = fileSize - (sizeof (TInt32) * 2);
239 User::LeaveIfError(file.Seek(ESeekStart,sizePos));
241 des.Set(REINTERPRET_CAST(TUint8 *,&aTrailer),0,sizeof(TInt32)*2);
243 User::LeaveIfError(file.Read(des));
245 CleanupStack::PopAndDestroy();
249 @deprecated Interface is deprecated because it is unsafe as it may leave. It is available for backward compatibility reasons only.
250 @see EZGZipFile::IsGzipFileL
252 EXPORT_C TBool EZGZipFile::IsGzipFile(RFs &aRfs, const TDesC &aFname)
254 TBool retBool = true;
255 TRAPD(errCode, retBool = IsGzipFileL(aRfs, aFname));
256 if(errCode != KErrNone)
264 Determine if the given file is a valid zip file
266 @param aRfs file server session
267 @param aFname name of the file to check
268 @leave ... Any of the system wide error codes
269 @return ETrue if the file is valid zip file, EFalse otherwise
271 EXPORT_C TBool EZGZipFile::IsGzipFileL(RFs &aRfs, const TDesC &aFname)
276 User::LeaveIfError(file.Open(aRfs,aFname,EFileStream | EFileRead | EFileShareAny));
277 CleanupClosePushL(file);
279 TPtr8 des(ids,0,sizeof(TUint8) * 2);
281 User::LeaveIfError(file.Read(des));
282 CleanupStack::PopAndDestroy();
283 return (ids[0] == ID1 && ids[1] == ID2);
286 //--------------------------------------------------------------------------------------------------------
289 CEZFileToGzipBM::CEZFileToGzipBM(RFile &aInput, RFile &aOutput) : CEZFileBufferManager(aInput,aOutput)
291 iCrc = crc32_r(iCrc,NULL,0);
295 CEZFileToGzipBM* CEZFileToGzipBM::NewLC(RFile &aInput, RFile &aOutput, TInt aBufferSize)
297 CEZFileToGzipBM *bm = new (ELeave) CEZFileToGzipBM(aInput,aOutput);
298 CleanupStack::PushL(bm);
299 bm->ConstructL(aBufferSize);
303 CEZFileToGzipBM* CEZFileToGzipBM::NewL(RFile &aInput, RFile &aOutput, TInt aBufferSize)
305 CEZFileToGzipBM *bm = new (ELeave) CEZFileToGzipBM(aInput,aOutput);
306 CleanupStack::PushL(bm);
307 bm->ConstructL(aBufferSize);
312 void CEZFileToGzipBM::NeedInputL(CEZZStream &aZStream)
314 CEZFileBufferManager::NeedInputL(aZStream);
315 iCrc = crc32_r(iCrc,iInputDescriptor.Ptr(),iInputDescriptor.Size());
318 void CEZFileToGzipBM::InitializeL(CEZZStream &aZStream)
320 CEZFileBufferManager::InitializeL(aZStream);
321 iCrc = crc32_r(iCrc,iInputDescriptor.Ptr(),iInputDescriptor.Size());
324 //--------------------------------------------------------------------------------------------------------
327 CEZGzipToFileBM::CEZGzipToFileBM(RFile &aInput, RFile &aOutput) : CEZFileBufferManager(aInput,aOutput)
329 iCrc = crc32_r(iCrc,NULL,0);
332 CEZGzipToFileBM* CEZGzipToFileBM::NewLC(RFile &aInput, RFile &aOutput, TInt aBufferSize)
334 CEZGzipToFileBM *bm = new (ELeave) CEZGzipToFileBM(aInput,aOutput);
335 CleanupStack::PushL(bm);
336 bm->ConstructL(aBufferSize);
340 CEZGzipToFileBM* CEZGzipToFileBM::NewL(RFile &aInput, RFile &aOutput, TInt aBufferSize)
342 CEZGzipToFileBM *bm = new (ELeave) CEZGzipToFileBM(aInput,aOutput);
343 CleanupStack::PushL(bm);
344 bm->ConstructL(aBufferSize);
349 void CEZGzipToFileBM::NeedOutputL(CEZZStream &aZStream)
351 TPtrC8 od = aZStream.OutputDescriptor();
352 iCrc = crc32_r(iCrc,od.Ptr(),od.Size());
353 CEZFileBufferManager::NeedOutputL(aZStream);
356 void CEZGzipToFileBM::FinalizeL(CEZZStream &aZStream)
358 TPtrC8 od = aZStream.OutputDescriptor();
359 iCrc = crc32_r(iCrc,od.Ptr(),od.Size());
360 CEZFileBufferManager::FinalizeL(aZStream);
364 //--------------------------------------------------------------------------------------------------------
365 CEZGZipToFile::CEZGZipToFile() : iDecompressor(NULL), iBufferManager(NULL)
370 CEZGZipToFile::~CEZGZipToFile()
372 delete iDecompressor;
373 delete iBufferManager;
378 Creates a new CEZGZipToFile object and leaves it on the CleanupStack
380 @param aRfs open file server session
381 @param aGzFileName name of the file to be de-compressed
382 @param aOutput the target file to hold the un-compressed data
383 @param aBufferSize required size of buffers
384 @return a pointer to the new CEZGZipToFile object, left on the CleanupStack
386 EXPORT_C CEZGZipToFile* CEZGZipToFile::NewLC(RFs &aRfs, const TDesC &aGzFileName, RFile &aOutput, TInt aBufferSize)
388 CEZGZipToFile* dec = new (ELeave) CEZGZipToFile;
389 CleanupStack::PushL(dec);
390 dec->ConstructL(aRfs,aGzFileName,aOutput,aBufferSize);
395 Creates a new CEZGZipToFile object
397 @param aRfs open file server session
398 @param aGzFileName name of the file to be de-compressed
399 @param aOutput the target file to hold the un-compressed data
400 @param aBufferSize required size of buffers
401 @return a pointer to the new CEZGZipToFile object
403 EXPORT_C CEZGZipToFile* CEZGZipToFile::NewL(RFs &aRfs, const TDesC &aGzFileName, RFile &aOutput, TInt aBufferSize)
405 CEZGZipToFile* dec = new (ELeave) CEZGZipToFile;
406 CleanupStack::PushL(dec);
407 dec->ConstructL(aRfs,aGzFileName,aOutput,aBufferSize);
413 Quits the current de-compression operation and restarts with the specified arguments
415 @param aRfs open file server session
416 @param aGzFileName name of the file to be de-compressed
417 @param aOutput the target file to hold the un-compressed data
418 @param aBufferSize required size of buffers
419 @leave ... Any of the system wide error codes
421 EXPORT_C void CEZGZipToFile::ResetL(RFs &aRfs, const TDesC &aGzFileName, RFile &aOutput, TInt aBufferSize)
423 delete iBufferManager;
424 iBufferManager = NULL;
425 InitialiseBufManL(aRfs,aGzFileName,aOutput,aBufferSize);
426 iDecompressor->ResetL(*iBufferManager);
430 De-compresses the current zip file in stages. The function needs to called again until the de-compression
431 is finalised, in which case it will return EFalse - for example...
434 while ( decompressor->InflateL() )
436 // No action required
440 @leave KEZlibErrBadGZipCrc Invalid CRC check
441 @leave ... Any of the system wide error codes
442 @return ETrue if the de-compression is not complete, and function must be called again
443 @return EFalse if the de-compression is finalised
445 EXPORT_C TBool CEZGZipToFile::InflateL()
447 TBool keepGoing = iDecompressor->InflateL();
451 if (iBufferManager->Crc() != iTrailer.iCrc32)
453 User::Leave(KEZlibErrBadGZipCrc);
462 void CEZGZipToFile::InitialiseBufManL(RFs &aRfs, const TDesC &aGzFileName, RFile &aOutput, TInt aBufferSize)
464 EZGZipFile::LocateAndReadTrailerL(aRfs,aGzFileName,iTrailer);
465 User::LeaveIfError(iGZipFile.Open(aRfs,aGzFileName,EFileStream | EFileRead | EFileShareAny));
466 EZGZipFile::ReadHeaderL(iGZipFile,iHeader);
467 iBufferManager = CEZGzipToFileBM::NewL(iGZipFile,aOutput,aBufferSize);
470 void CEZGZipToFile::ConstructL(RFs &aRfs, const TDesC &aGzFileName, RFile &aOutput, TInt aBufferSize)
472 InitialiseBufManL(aRfs,aGzFileName,aOutput,aBufferSize);
474 // this is a special zlib modification to stop it choking when it can't find the normal zlib stream header.
476 iDecompressor = CEZDecompressor::NewL(*iBufferManager,-CEZDecompressor::EMaxWBits);
480 //--------------------------------------------------------------------------------------------------------
483 CEZFileToGZip::CEZFileToGZip() : iCompressor(NULL), iBufferManager(NULL)
488 CEZFileToGZip::~CEZFileToGZip()
491 delete iBufferManager;
496 Creates a new CEZFileToGZip object and leaves it on the CleanupStack
498 @param aRfs open file server session
499 @param aGzFileName the name of the target zip file
500 @param aInput the file to compress
501 @param aBufferSize required size of buffers
502 @return a pointer to the new CEZFileToGZip object, left on the CleanupStack
504 EXPORT_C CEZFileToGZip* CEZFileToGZip::NewLC(RFs &aRfs, const TDesC &aGzFileName, RFile &aInput, TInt aBufferSize)
506 CEZFileToGZip* com = new (ELeave) CEZFileToGZip;
507 CleanupStack::PushL(com);
508 com->ConstructL(aRfs,aGzFileName,aInput,aBufferSize);
513 Creates a new CEZFileToGZip object and leaves it on the CleanupStack
515 @param aRfs open file server session
516 @param aGzFileName the name of the target zip file
517 @param aInput the file to compress
518 @param aBufferSize required size of buffers
519 @return a pointer to the new CEZFileToGZip object, left on the CleanupStack
521 EXPORT_C CEZFileToGZip* CEZFileToGZip::NewL(RFs &aRfs, const TDesC &aGzFileName, RFile &aInput, TInt aBufferSize)
523 CEZFileToGZip* com = new (ELeave) CEZFileToGZip;
524 CleanupStack::PushL(com);
525 com->ConstructL(aRfs,aGzFileName,aInput,aBufferSize);
531 Quits the current compression operation and restarts with the specified arguments
533 @param aRfs open file server session
534 @param aGzFileName the name of the target zip file
535 @param aInput the file to compress
536 @param aBufferSize required size of buffers
537 @leave ... Any of the system wide error codes
539 EXPORT_C void CEZFileToGZip::ResetL(RFs &aRfs, const TDesC &aGzFileName, RFile &aInput, TInt aBufferSize)
541 delete iBufferManager;
542 iBufferManager = NULL;
543 InitialiseBufManL(aRfs,aGzFileName,aInput,aBufferSize);
544 iCompressor->ResetL(*iBufferManager);
548 Compresses the current file in stages. The function needs to called again until the compression
549 is finalised, in which case it will return EFalse - for example...
552 while ( compressor->DeflateL() )
554 // No action required
558 @leave ... Any of the system wide error codes
559 @return ETrue if the compression is not complete, and function must be called again
560 @return EFalse if the compression is finalised
562 EXPORT_C TBool CEZFileToGZip::DeflateL()
564 TBool keepgoing = iCompressor->DeflateL();
567 TEZGZipTrailer trailer(iBufferManager->Crc(), iUncompressedDataSize);
568 EZGZipFile::WriteTrailerL(iGZipFile, trailer);
575 void CEZFileToGZip::InitialiseBufManL(RFs &aRfs, const TDesC &aGzFileName, RFile &aInput, TInt aBufferSize)
577 User::LeaveIfError(aInput.Size(iUncompressedDataSize));
580 err = iGZipFile.Create(aRfs, aGzFileName,EFileStream | EFileWrite | EFileShareExclusive);
581 if (err == KErrAlreadyExists)
582 User::LeaveIfError(iGZipFile.Open(aRfs,aGzFileName,EFileStream | EFileWrite | EFileShareExclusive));
584 User::LeaveIfError(err);
586 EZGZipFile::WriteHeaderL(iGZipFile,iHeader);
587 iBufferManager = CEZFileToGzipBM::NewL(aInput,iGZipFile,aBufferSize);
590 void CEZFileToGZip::ConstructL(RFs &aRfs, const TDesC &aGzFileName, RFile &aInput, TInt aBufferSize)
592 InitialiseBufManL(aRfs,aGzFileName,aInput,aBufferSize);
594 // this is a special zlib modification to stop zlib writing out its normal zlib stream header.
596 iCompressor = CEZCompressor::NewL(*iBufferManager,CEZCompressor::EDefaultCompression,-CEZCompressor::EMaxWBits);