1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/ossrv/compressionlibs/ziplib/src/ezlib/compressor.cpp Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,314 @@
1.4 +// Copyright (c) 2003-2009 Nokia Corporation and/or its subsidiary(-ies).
1.5 +// All rights reserved.
1.6 +// This component and the accompanying materials are made available
1.7 +// under the terms of "Eclipse Public License v1.0"
1.8 +// which accompanies this distribution, and is available
1.9 +// at the URL "http://www.eclipse.org/legal/epl-v10.html".
1.10 +//
1.11 +// Initial Contributors:
1.12 +// Nokia Corporation - initial contribution.
1.13 +//
1.14 +// Contributors:
1.15 +//
1.16 +// Description:
1.17 +//
1.18 +
1.19 +#include <ezcompressor.h>
1.20 +#include "libzcore.h"
1.21 +
1.22 +CEZCompressor::CEZCompressor(MEZBufferManager* aInit) :
1.23 + iBufferInit(aInit)
1.24 + {
1.25 +
1.26 + }
1.27 +
1.28 +CEZCompressor::~CEZCompressor()
1.29 + {
1.30 + // Note that deflateEnd may have already been called by zlib if for example and alloc failure
1.31 + // occurred in deflateInit2. However there is no harm in calling deflateEnd twice.
1.32 +
1.33 + deflateEnd_r(&iStream);
1.34 + }
1.35 +
1.36 +/**
1.37 +Creates a new CEZCompressor object and leaves it on the CleanupStack
1.38 +
1.39 +@param aInit buffer manager to handle both input and output buffers
1.40 +@param aLevel compression levels
1.41 +@param aWindowBits the base two logarithm of the window size (the size of the history buffer). It should
1.42 +be in the range 8..15 for this version of the library. Larger values of this parameter result in better
1.43 +compression at the expense of memory usage.
1.44 +@param aMemLevel specifies how much memory should be allocated for the internal compression state.
1.45 +memLevel=1 uses minimum memory but is slow and reduces compression ratio; memLevel=9 uses maximum memory
1.46 +for optimal speed.
1.47 +@param aStrategy compression strategy - used to tune the compression algorithm. The strategy parameter only affects
1.48 +the compression ratio but not the correctness of the compressed output even if it is not set appropriately
1.49 +@see TStrategy
1.50 +@return the new CEZCompressor object (on the CleanupStack)
1.51 +*/
1.52 +EXPORT_C CEZCompressor *CEZCompressor::NewLC(MEZBufferManager& aInit, TInt aLevel, TInt aWindowBits,
1.53 + TInt aMemLevel, TStrategy aStrategy)
1.54 + {
1.55 + CEZCompressor* deflater = new (ELeave) CEZCompressor(&aInit);
1.56 + CleanupStack::PushL(deflater);
1.57 + deflater->ConstructL(aLevel,aWindowBits,aMemLevel,aStrategy);
1.58 + return deflater;
1.59 + }
1.60 +
1.61 +/**
1.62 +Creates a new CEZCompressor object
1.63 +
1.64 +@param aInit buffer manager to handle both input and output buffers
1.65 +@param aLevel compression levels
1.66 +@param aWindowBits the base two logarithm of the window size (the size of the history buffer). It should
1.67 +be in the range 8..15 for this version of the library. Larger values of this parameter result in better
1.68 +compression at the expense of memory usage.
1.69 +@param aMemLevel specifies how much memory should be allocated for the internal compression state.
1.70 +memLevel=1 uses minimum memory but is slow and reduces compression ratio; memLevel=9 uses maximum memory
1.71 +for optimal speed.
1.72 +@param aStrategy compression strategy - used to tune the compression algorithm. The strategy parameter only affects
1.73 +the compression ratio but not the correctness of the compressed output even if it is not set appropriately
1.74 +@see TStrategy
1.75 +@return the new CEZCompressor object
1.76 +*/
1.77 +EXPORT_C CEZCompressor* CEZCompressor::NewL(MEZBufferManager& aInit, TInt aLevel, TInt aWindowBits,
1.78 + TInt aMemLevel, TStrategy aStrategy)
1.79 + {
1.80 + CEZCompressor* deflater = new (ELeave) CEZCompressor(&aInit);
1.81 + CleanupStack::PushL(deflater);
1.82 + deflater->ConstructL(aLevel,aWindowBits,aMemLevel,aStrategy);
1.83 + CleanupStack::Pop();
1.84 + return deflater;
1.85 + }
1.86 +
1.87 +/**
1.88 +Overload of CEZCompressor constructor takes aDictionary argument
1.89 +
1.90 +@param aInit buffer manager to handle both input and output buffers
1.91 +@param aDictionary used to initialize the compression dictionary from the given byte sequence
1.92 +without producing any compressed output. The compressor and decompressor must use exactly the same dictionary.
1.93 +The dictionary should consist of strings (byte sequences) that are likely to be encountered later in the data to be compressed,
1.94 +with the most commonly used strings preferably put towards the end of the dictionary. Using a dictionary is most useful
1.95 +when the data to be compressed is short and can be predicted with good accuracy; the data can then be compressed better than
1.96 +with the default empty dictionary.
1.97 +@param aLevel compression level
1.98 +@param aWindowBits the base two logarithm of the window size (the size of the history buffer). It should
1.99 +be in the range 8..15 for this version of the library. Larger values of this parameter result in better
1.100 +compression at the expense of memory usage.
1.101 +@param aMemLevel specifies how much memory should be allocated for the internal compression state.
1.102 +memLevel=1 uses minimum memory but is slow and reduces compression ratio; memLevel=9 uses maximum memory
1.103 +for optimal speed.
1.104 +@param aStrategy compression strategy - used to tune the compression algorithm. The strategy parameter only affects
1.105 +the compression ratio but not the correctness of the compressed output even if it is not set appropriately
1.106 +@see TStrategy
1.107 +@return the new CEZCompressor object (on the CleanupStack)
1.108 +*/
1.109 +EXPORT_C CEZCompressor* CEZCompressor::NewLC(MEZBufferManager& aInit, const TDesC8& aDictionary,
1.110 + TInt aLevel, TInt aWindowBits, TInt aMemLevel, TStrategy aStrategy)
1.111 + {
1.112 + CEZCompressor* deflater = new (ELeave) CEZCompressor(&aInit);
1.113 + CleanupStack::PushL(deflater);
1.114 + deflater->ConstructL(aLevel,aDictionary.Ptr(),aDictionary.Size(),aWindowBits,aMemLevel,aStrategy);
1.115 + return deflater;
1.116 + }
1.117 +
1.118 +/**
1.119 +Overload of CEZCompressor constructor takes aDictionary argument
1.120 +
1.121 +@param aInit buffer manager to handle both input and output buffers
1.122 +@param aDictionary used to initialize the compression dictionary from the given byte sequence
1.123 +without producing any compressed output. The compressor and decompressor must use exactly the same dictionary.
1.124 +The dictionary should consist of strings (byte sequences) that are likely to be encountered later in the data to be compressed,
1.125 +with the most commonly used strings preferably put towards the end of the dictionary. Using a dictionary is most useful
1.126 +when the data to be compressed is short and can be predicted with good accuracy; the data can then be compressed better than
1.127 +with the default empty dictionary.
1.128 +@param aLevel compression level
1.129 +@param aWindowBits the base two logarithm of the window size (the size of the history buffer). It should
1.130 +be in the range 8..15 for this version of the library. Larger values of this parameter result in better
1.131 +compression at the expense of memory usage.
1.132 +@param aMemLevel specifies how much memory should be allocated for the internal compression state.
1.133 +memLevel=1 uses minimum memory but is slow and reduces compression ratio; memLevel=9 uses maximum memory
1.134 +for optimal speed.
1.135 +@param aStrategy compression strategy - used to tune the compression algorithm. The strategy parameter only affects
1.136 +the compression ratio but not the correctness of the compressed output even if it is not set appropriately
1.137 +@see TStrategy
1.138 +@return the new CEZCompressor object
1.139 +*/
1.140 +EXPORT_C CEZCompressor* CEZCompressor::NewL(MEZBufferManager& aInit, const TDesC8& aDictionary,
1.141 + TInt aLevel, TInt aWindowBits, TInt aMemLevel, TStrategy aStrategy)
1.142 + {
1.143 + CEZCompressor* deflater = new (ELeave) CEZCompressor(&aInit);
1.144 + CleanupStack::PushL(deflater);
1.145 + deflater->ConstructL(aLevel,aDictionary.Ptr(),aDictionary.Size(),aWindowBits,aMemLevel,aStrategy);
1.146 + CleanupStack::Pop();
1.147 + return deflater;
1.148 + }
1.149 +
1.150 +/**
1.151 +Resets the current compression operation, with the new buffer manager
1.152 +
1.153 +@param aInit new buffer manager to handle the new input and output buffers
1.154 +@leave ... Any of the system wide error codes
1.155 +*/
1.156 +EXPORT_C void CEZCompressor::ResetL(MEZBufferManager& aInit)
1.157 + {
1.158 + iBufferInit = &aInit;
1.159 + iBufferInit->InitializeL(*this);
1.160 + if (deflateReset_r(&iStream) == Z_STREAM_ERROR)
1.161 + User::Leave(KEZlibErrStream);
1.162 + iDeflationState = ENoFlush;
1.163 + }
1.164 +
1.165 +/**
1.166 +Compress the data to the buffer in stages, return value indicates if the compression has finalised
1.167 +or if further calls are necessary
1.168 +
1.169 +@leave KEZlibErrStream There is a problem with the stream
1.170 +@leave KEZlibErrBuf There is a problem with the buffer
1.171 +@leave KEZlibErrUnexpected Unexpected programming error
1.172 +@leave ... Any of the System wide error codes
1.173 +@return ETrue if the function must be called again, EFalse if compression is finalised
1.174 +*/
1.175 +EXPORT_C TBool CEZCompressor::DeflateL()
1.176 + {
1.177 + TInt err;
1.178 + TBool callAgain = ETrue;
1.179 +
1.180 + switch (iDeflationState)
1.181 + {
1.182 + case ENoFlush:
1.183 + err = deflate_r(&iStream,Z_NO_FLUSH);
1.184 +
1.185 + switch (err)
1.186 + {
1.187 + case Z_STREAM_ERROR:
1.188 + User::Leave(KEZlibErrStream);
1.189 + break;
1.190 +
1.191 + case Z_OK:
1.192 + if (iStream.avail_in == 0)
1.193 + iBufferInit->NeedInputL(*this);
1.194 +
1.195 + if (iStream.avail_out == 0)
1.196 + iBufferInit->NeedOutputL(*this);
1.197 + break;
1.198 +
1.199 + case Z_BUF_ERROR:
1.200 + // this is probably ok we have just run out of input.
1.201 +
1.202 + iDeflationState = EFinish;
1.203 + break;
1.204 +
1.205 + default:
1.206 +
1.207 + // there's something wrong with this code if we get here !
1.208 +
1.209 + User::Leave(KEZlibErrUnexpected);
1.210 + break;
1.211 + }
1.212 +
1.213 + break;
1.214 +
1.215 + case EFinish:
1.216 + err = deflate_r(&iStream,Z_FINISH);
1.217 +
1.218 + switch (err)
1.219 + {
1.220 + case Z_STREAM_ERROR:
1.221 + User::Leave(KEZlibErrStream);
1.222 + break;
1.223 +
1.224 + case Z_BUF_ERROR:
1.225 + User::Leave(KEZlibErrBuf);
1.226 + break;
1.227 +
1.228 + case Z_OK:
1.229 + if (iStream.avail_in == 0)
1.230 + iBufferInit->NeedInputL(*this);
1.231 +
1.232 + if (iStream.avail_out == 0)
1.233 + iBufferInit->NeedOutputL(*this);
1.234 + break;
1.235 +
1.236 + case Z_STREAM_END:
1.237 + iDeflationState = EFinalize;
1.238 + break;
1.239 +
1.240 + default:
1.241 + // there's something wrong with this code if we get here !
1.242 +
1.243 + User::Leave(KEZlibErrUnexpected);
1.244 + break;
1.245 + }
1.246 +
1.247 + break;
1.248 +
1.249 + case EFinalize:
1.250 + iBufferInit->FinalizeL(*this);
1.251 + callAgain = EFalse;
1.252 + iDeflationState = ETerminated;
1.253 + break;
1.254 +
1.255 + case ETerminated:
1.256 + User::Leave(KEZlibErrDeflateTerminated);
1.257 + }
1.258 +
1.259 + return callAgain;
1.260 + }
1.261 +
1.262 +void CEZCompressor::ConstructL(TInt aLevel, const TUint8 *aDictionary, TInt aLength,
1.263 + TInt aWindowBits, TInt aMemLevel, TStrategy aStrategy)
1.264 + {
1.265 + ConstructL(aLevel,aWindowBits,aMemLevel,aStrategy);
1.266 + if (deflateSetDictionary_r(&iStream,STATIC_CAST(const Bytef *,aDictionary),aLength) ==
1.267 + Z_STREAM_ERROR)
1.268 + User::Leave(KEZlibErrStream); // This should never happen.
1.269 + }
1.270 +
1.271 +void CEZCompressor::ConstructL(TInt aLevel, TInt aWindowBits, TInt aMemLevel, TStrategy aStrategy)
1.272 + {
1.273 + // don't need to assert the validity of aWindowBits, aMemLevel & aStrategy as deflateInit2 will
1.274 + // do this for us.
1.275 +
1.276 + iStream.zalloc = Z_NULL;
1.277 + iStream.zfree = Z_NULL;
1.278 + iStream.opaque = Z_NULL;
1.279 +
1.280 + iBufferInit->InitializeL(*this);
1.281 +
1.282 + TInt err = deflateInit2_r(&iStream,aLevel,Z_DEFLATED,aWindowBits,aMemLevel, STATIC_CAST(int,aStrategy));
1.283 + if (err == Z_STREAM_ERROR)
1.284 + User::Leave(KEZlibErrStream);
1.285 + else if (err == Z_MEM_ERROR)
1.286 + User::LeaveNoMemory();
1.287 +
1.288 + iDeflationState = ENoFlush;
1.289 + }
1.290 +
1.291 +/**
1.292 +Compresses the data in the given buffer
1.293 +
1.294 +@param aDestination the target buffer for the compressed data
1.295 +@param aSource the buffer containing the data to be compressed
1.296 +@param aLevel the level of compression
1.297 +@leave KEZLibErrBuf There is a problem with the buffer
1.298 +@leave KEZLIbErrStream There is a problem with the stream
1.299 +@leave ... Any of the system wide error codes
1.300 +*/
1.301 +EXPORT_C void CEZCompressor::CompressL(TDes8 &aDestination, const TDesC8 &aSource,
1.302 + TInt aLevel)
1.303 + {
1.304 + Bytef* destinationBuffer = STATIC_CAST(Bytef* ,CONST_CAST(TUint8* ,aDestination.Ptr()));
1.305 + const Bytef* sourceBuffer = STATIC_CAST(const Bytef* ,aSource.Ptr());
1.306 + uLongf dl = aDestination.MaxSize();
1.307 + TInt err = compress2_r(destinationBuffer,&dl,sourceBuffer,aSource.Size(),aLevel);
1.308 +
1.309 + if (err == Z_MEM_ERROR)
1.310 + User::LeaveNoMemory();
1.311 + else if (err == Z_BUF_ERROR)
1.312 + User::Leave(KEZlibErrBuf);
1.313 + else if (err == Z_STREAM_ERROR)
1.314 + User::Leave(KEZlibErrStream);
1.315 +
1.316 + aDestination.SetLength(dl);
1.317 + }