First public contribution.
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.
18 #include <zipfilememberinputstream.h>
21 RZipFileMemberReaderStream::RZipFileMemberReaderStream(
24 TUint32 aCompressedSize,
25 TUint32 aUncompressedSize,
26 TUint32 aCompressionMethod):
28 iCompressionMethod(aCompressionMethod),
29 iCompressedSize(aCompressedSize),
30 iUncompressedSize(aUncompressedSize),
31 iFileOffset(aDataOffset)
36 RZipFileMemberReaderStream* RZipFileMemberReaderStream::NewL(
39 TUint32 aCompressedSize,
40 TUint32 aUncompressedSize,
41 TUint32 aCompressionMethod)
43 RZipFileMemberReaderStream* me = new(ELeave) RZipFileMemberReaderStream(aZipFile, aDataOffset, aCompressedSize,aUncompressedSize, aCompressionMethod);
44 CleanupStack::PushL(me);
46 CleanupStack::Pop(me);
51 Creates input stream to be used for reading the contents of the compressed file.
53 void RZipFileMemberReaderStream::ConstructL()
55 TInt err = inflateInit2_r(&iStream, -MAX_WBITS);
56 if (err == Z_MEM_ERROR)
58 User::Leave(KErrNoMemory);
63 Destructor. All dynamically allocated data structures for this stream are freed.
65 EXPORT_C RZipFileMemberReaderStream::~RZipFileMemberReaderStream()
67 inflateEnd_r(&iStream);
71 Reads data from the stream buffer into the specified descriptor.
72 On return, contains the data read from the stream buffer
74 @param aDes The target descriptor for the data read from the stream buffer
75 @param aLength The maximum number of bytes to be read
76 @return KErrNone If all bytes read successfully.
77 @return KErrCorrupt If reading fails.
78 @return KErrEof If end of file is reached.
79 @return ... Any one of the system-wide error codes for other errors.
81 EXPORT_C TInt RZipFileMemberReaderStream::Read(TDes16& aDes, TInt aLength)
83 TUint32 numBytesRead = 0;
84 TInt err = Read(CONST_CAST(TByte*,(const TByte*)aDes.Ptr()), 2*aLength, &numBytesRead);
87 aDes.SetLength( (err==KErrEof) ? numBytesRead>>2 : 0 );
91 aDes.SetLength(numBytesRead>>2);
95 TInt RZipFileMemberReaderStream::Read(void)
100 if (Read(&b, 1, &numRead) == 1)
110 TInt RZipFileMemberReaderStream::Read(TDes8& aDes, TInt aLength)
112 TUint32 numBytesRead = 0;
113 TInt err = Read(CONST_CAST(TByte*,aDes.Ptr()), aLength, &numBytesRead);
116 aDes.SetLength( (err==KErrEof) ? numBytesRead : 0 );
120 aDes.SetLength(numBytesRead);
124 void RZipFileMemberReaderStream::ReadL(TDes16& aDes, TInt aLength)
126 User::LeaveIfError(Read(aDes, aLength));
129 void RZipFileMemberReaderStream::Release()
132 void RZipFileMemberReaderStream::Close()
136 TInt RZipFileMemberReaderStream::Read(TByte* aBytes, TUint32 aLength, TUint32* aRetLength)
138 if (iCompressionMethod == CZipArchive::EDeflated)
140 return GetBytes(aBytes, aLength, aRetLength);
144 return GetStoredBytes(aBytes, aLength, aRetLength);
148 TInt RZipFileMemberReaderStream::GetBytes(TByte* aBytes, TUint32 aLength, TUint32* aRetLength)
150 TUint32 bytesLeftToRead;
155 // Be careful not to confuse compressed bytes and uncompressed bytes.
156 // The length passed in is in uncompressed bytes, compressed bytes
157 // are mainly used in the GetCompressedBytes() function called.
158 // iBytesLength refers to the number of bytes already read.
159 // If the request is to read past the end of the file
160 // we should return KErrNone on the first instance, i.e. return all bytes
161 // read successfully. On the second instance return KErrEof, i.e. we have
162 // already read all the bytes when another request comes in, so return KErrEof.
163 // This follows the rules for reading an uncompressed file within this component
164 // and this is also the way that RFile::ReadL() does it.
166 if (aLength > iUncompressedSize)
168 aLength = iUncompressedSize; // no point trying to read more than we have
171 bytesLeftToRead = aLength;
173 while (bytesLeftToRead > 0)
175 if (iStream.avail_in == 0)
177 if (GetCompressedBytes() != KErrNone)
183 // The decompression will be done in the user provided memory.
184 iStream.next_out = &aBytes[iBytesLength];
185 iStream.avail_out = aLength - iBytesLength;
186 status = inflate_r(&iStream, Z_SYNC_FLUSH);
191 iBytesLength = aLength - iStream.avail_out;
194 case Z_STREAM_END: //EOF
195 if (iBytesLength == aLength - iStream.avail_out)
197 *aRetLength = iBytesLength;
202 iBytesLength = aLength - iStream.avail_out;
212 bytesLeftToRead = aLength - iBytesLength;
215 *aRetLength = iBytesLength;
219 TInt RZipFileMemberReaderStream::GetCompressedBytes(void)
221 if (iOffset < iCompressedSize)
224 TUint32 nBytesToRead;
226 nBytesLeft = iCompressedSize - iOffset;
227 nBytesToRead = (nBytesLeft > sizeof(iCompressedBytes)) ? sizeof(iCompressedBytes) : nBytesLeft;
228 if (iZipFile.Seek(iFileOffset) != KErrNone)
233 if (iZipFile.Read(iCompressedBytes, nBytesToRead) != KErrNone)
237 iFileOffset += nBytesToRead;
238 iOffset += nBytesToRead;
239 iStream.next_in = iCompressedBytes;
240 iStream.avail_in = nBytesToRead;
246 iCompressedBytes[0] = 0;
247 iStream.avail_in = 1;
248 iStream.next_in = iCompressedBytes;
258 TInt RZipFileMemberReaderStream::GetStoredBytes(TByte* aBytes, TUint32 aLength, TUint32* aRetLength)
262 if (aLength > iUncompressedSize)
264 aLength = iUncompressedSize; // no point trying to read more than we have
266 if (aLength == 0) // empty file like a directory
270 if (iOffset == iCompressedSize) // end of zip item.
274 if ((iOffset + aLength) > iCompressedSize)
276 aLength = iCompressedSize - iOffset; // adjust read to what is left
282 if (iZipFile.Seek(iFileOffset) != KErrNone)
286 status = iZipFile.Read(aBytes, aLength);
287 if (status == KErrNone)
289 iFileOffset += aLength;
291 *aRetLength = aLength;