os/ossrv/compressionlibs/ziplib/src/ezlib/compressor.cpp
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
sl@0
     1
// Copyright (c) 2003-2009 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     2
// All rights reserved.
sl@0
     3
// This component and the accompanying materials are made available
sl@0
     4
// under the terms of "Eclipse Public License v1.0"
sl@0
     5
// which accompanies this distribution, and is available
sl@0
     6
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     7
//
sl@0
     8
// Initial Contributors:
sl@0
     9
// Nokia Corporation - initial contribution.
sl@0
    10
//
sl@0
    11
// Contributors:
sl@0
    12
//
sl@0
    13
// Description:
sl@0
    14
//
sl@0
    15
sl@0
    16
#include <ezcompressor.h>
sl@0
    17
#include "libzcore.h"
sl@0
    18
sl@0
    19
CEZCompressor::CEZCompressor(MEZBufferManager* aInit) : 
sl@0
    20
		iBufferInit(aInit)
sl@0
    21
	{
sl@0
    22
	
sl@0
    23
	}
sl@0
    24
sl@0
    25
CEZCompressor::~CEZCompressor()
sl@0
    26
	{
sl@0
    27
		// Note that deflateEnd may have already been called by zlib if for example and alloc failure
sl@0
    28
		// occurred in deflateInit2.  However there is no harm in calling deflateEnd twice.
sl@0
    29
sl@0
    30
		deflateEnd_r(&iStream);		
sl@0
    31
	}
sl@0
    32
sl@0
    33
/**
sl@0
    34
Creates a new CEZCompressor object and leaves it on the CleanupStack
sl@0
    35
sl@0
    36
@param aInit buffer manager to handle both input and output buffers
sl@0
    37
@param aLevel compression levels
sl@0
    38
@param aWindowBits the base two logarithm of the window size (the size of the history buffer).  It should 
sl@0
    39
be in the range 8..15 for this version of the library. Larger values of this parameter result in better 
sl@0
    40
compression at the expense of memory usage.
sl@0
    41
@param aMemLevel specifies how much memory should be allocated for the internal compression state.  
sl@0
    42
memLevel=1 uses minimum memory but is slow and reduces compression ratio; memLevel=9 uses maximum memory
sl@0
    43
for optimal speed.
sl@0
    44
@param aStrategy compression strategy - used to tune the compression algorithm.  The strategy parameter only affects 
sl@0
    45
the compression ratio but not the correctness of the compressed output even if it is not set appropriately
sl@0
    46
@see TStrategy
sl@0
    47
@return the new CEZCompressor object (on the CleanupStack)
sl@0
    48
*/
sl@0
    49
EXPORT_C CEZCompressor *CEZCompressor::NewLC(MEZBufferManager& aInit, TInt aLevel, TInt aWindowBits, 
sl@0
    50
									 TInt aMemLevel, TStrategy aStrategy)
sl@0
    51
	{
sl@0
    52
	CEZCompressor* deflater = new (ELeave) CEZCompressor(&aInit);
sl@0
    53
	CleanupStack::PushL(deflater);
sl@0
    54
	deflater->ConstructL(aLevel,aWindowBits,aMemLevel,aStrategy);
sl@0
    55
	return deflater;
sl@0
    56
	}
sl@0
    57
sl@0
    58
/**
sl@0
    59
Creates a new CEZCompressor object
sl@0
    60
sl@0
    61
@param aInit buffer manager to handle both input and output buffers
sl@0
    62
@param aLevel compression levels
sl@0
    63
@param aWindowBits the base two logarithm of the window size (the size of the history buffer).  It should 
sl@0
    64
be in the range 8..15 for this version of the library. Larger values of this parameter result in better 
sl@0
    65
compression at the expense of memory usage.
sl@0
    66
@param aMemLevel specifies how much memory should be allocated for the internal compression state.  
sl@0
    67
memLevel=1 uses minimum memory but is slow and reduces compression ratio; memLevel=9 uses maximum memory
sl@0
    68
for optimal speed.
sl@0
    69
@param aStrategy compression strategy - used to tune the compression algorithm.  The strategy parameter only affects 
sl@0
    70
the compression ratio but not the correctness of the compressed output even if it is not set appropriately
sl@0
    71
@see TStrategy
sl@0
    72
@return the new CEZCompressor object
sl@0
    73
*/
sl@0
    74
EXPORT_C CEZCompressor* CEZCompressor::NewL(MEZBufferManager& aInit, TInt aLevel, TInt aWindowBits, 
sl@0
    75
									TInt aMemLevel, TStrategy aStrategy)
sl@0
    76
	{
sl@0
    77
	CEZCompressor* deflater = new (ELeave) CEZCompressor(&aInit);
sl@0
    78
	CleanupStack::PushL(deflater);
sl@0
    79
	deflater->ConstructL(aLevel,aWindowBits,aMemLevel,aStrategy);
sl@0
    80
	CleanupStack::Pop();
sl@0
    81
	return deflater;
sl@0
    82
	}
sl@0
    83
sl@0
    84
/**
sl@0
    85
Overload of CEZCompressor constructor takes aDictionary argument
sl@0
    86
sl@0
    87
@param aInit buffer manager to handle both input and output buffers
sl@0
    88
@param aDictionary used to initialize the compression dictionary from the given byte sequence
sl@0
    89
without producing any compressed output.  The compressor and decompressor must use exactly the same dictionary.  
sl@0
    90
The dictionary should consist of strings (byte sequences) that are likely to be encountered later in the data to be compressed, 
sl@0
    91
with the most commonly used strings preferably put towards the end of the dictionary. Using a dictionary is most useful 
sl@0
    92
when the data to be compressed is short and can be predicted with good accuracy; the data can then be compressed better than 
sl@0
    93
with the default empty dictionary.
sl@0
    94
@param aLevel compression level
sl@0
    95
@param aWindowBits the base two logarithm of the window size (the size of the history buffer).  It should 
sl@0
    96
be in the range 8..15 for this version of the library. Larger values of this parameter result in better 
sl@0
    97
compression at the expense of memory usage.
sl@0
    98
@param aMemLevel specifies how much memory should be allocated for the internal compression state.  
sl@0
    99
memLevel=1 uses minimum memory but is slow and reduces compression ratio; memLevel=9 uses maximum memory
sl@0
   100
for optimal speed.
sl@0
   101
@param aStrategy compression strategy - used to tune the compression algorithm.  The strategy parameter only affects 
sl@0
   102
the compression ratio but not the correctness of the compressed output even if it is not set appropriately
sl@0
   103
@see TStrategy
sl@0
   104
@return the new CEZCompressor object (on the CleanupStack)
sl@0
   105
*/
sl@0
   106
EXPORT_C CEZCompressor* CEZCompressor::NewLC(MEZBufferManager& aInit, const TDesC8& aDictionary, 
sl@0
   107
				TInt aLevel, TInt aWindowBits, TInt aMemLevel, TStrategy aStrategy)
sl@0
   108
	{
sl@0
   109
	CEZCompressor* deflater = new (ELeave) CEZCompressor(&aInit);
sl@0
   110
	CleanupStack::PushL(deflater);
sl@0
   111
	deflater->ConstructL(aLevel,aDictionary.Ptr(),aDictionary.Size(),aWindowBits,aMemLevel,aStrategy);
sl@0
   112
	return deflater;	
sl@0
   113
	}
sl@0
   114
sl@0
   115
/**
sl@0
   116
Overload of CEZCompressor constructor takes aDictionary argument
sl@0
   117
sl@0
   118
@param aInit buffer manager to handle both input and output buffers
sl@0
   119
@param aDictionary used to initialize the compression dictionary from the given byte sequence
sl@0
   120
without producing any compressed output.  The compressor and decompressor must use exactly the same dictionary.  
sl@0
   121
The dictionary should consist of strings (byte sequences) that are likely to be encountered later in the data to be compressed, 
sl@0
   122
with the most commonly used strings preferably put towards the end of the dictionary. Using a dictionary is most useful 
sl@0
   123
when the data to be compressed is short and can be predicted with good accuracy; the data can then be compressed better than 
sl@0
   124
with the default empty dictionary.   
sl@0
   125
@param aLevel compression level
sl@0
   126
@param aWindowBits the base two logarithm of the window size (the size of the history buffer).  It should 
sl@0
   127
be in the range 8..15 for this version of the library. Larger values of this parameter result in better 
sl@0
   128
compression at the expense of memory usage.
sl@0
   129
@param aMemLevel specifies how much memory should be allocated for the internal compression state.  
sl@0
   130
memLevel=1 uses minimum memory but is slow and reduces compression ratio; memLevel=9 uses maximum memory
sl@0
   131
for optimal speed.
sl@0
   132
@param aStrategy compression strategy - used to tune the compression algorithm.  The strategy parameter only affects 
sl@0
   133
the compression ratio but not the correctness of the compressed output even if it is not set appropriately
sl@0
   134
@see TStrategy
sl@0
   135
@return the new CEZCompressor object
sl@0
   136
*/
sl@0
   137
EXPORT_C CEZCompressor* CEZCompressor::NewL(MEZBufferManager& aInit, const TDesC8& aDictionary,  
sl@0
   138
			TInt aLevel, TInt aWindowBits, TInt aMemLevel, TStrategy aStrategy)
sl@0
   139
	{
sl@0
   140
	CEZCompressor* deflater = new (ELeave) CEZCompressor(&aInit);
sl@0
   141
	CleanupStack::PushL(deflater);
sl@0
   142
	deflater->ConstructL(aLevel,aDictionary.Ptr(),aDictionary.Size(),aWindowBits,aMemLevel,aStrategy);
sl@0
   143
	CleanupStack::Pop();
sl@0
   144
	return deflater;	
sl@0
   145
	}
sl@0
   146
sl@0
   147
/**
sl@0
   148
Resets the current compression operation, with the new buffer manager
sl@0
   149
sl@0
   150
@param aInit new buffer manager to handle the new input and output buffers
sl@0
   151
@leave ... Any of the system wide error codes
sl@0
   152
*/
sl@0
   153
EXPORT_C void CEZCompressor::ResetL(MEZBufferManager& aInit)
sl@0
   154
	{
sl@0
   155
	iBufferInit = &aInit;
sl@0
   156
	iBufferInit->InitializeL(*this);
sl@0
   157
	if (deflateReset_r(&iStream) == Z_STREAM_ERROR)		
sl@0
   158
		User::Leave(KEZlibErrStream);
sl@0
   159
	iDeflationState = ENoFlush;
sl@0
   160
	}
sl@0
   161
sl@0
   162
/**
sl@0
   163
Compress the data to the buffer in stages, return value indicates if the compression has finalised 
sl@0
   164
or if further calls are necessary
sl@0
   165
sl@0
   166
@leave KEZlibErrStream There is a problem with the stream
sl@0
   167
@leave KEZlibErrBuf There is a problem with the buffer
sl@0
   168
@leave KEZlibErrUnexpected Unexpected programming error
sl@0
   169
@leave ... Any of the System wide error codes
sl@0
   170
@return ETrue if the function must be called again, EFalse if compression is finalised
sl@0
   171
*/
sl@0
   172
EXPORT_C TBool CEZCompressor::DeflateL()
sl@0
   173
	{
sl@0
   174
	TInt err;
sl@0
   175
	TBool callAgain = ETrue;
sl@0
   176
	
sl@0
   177
	switch (iDeflationState)
sl@0
   178
		{
sl@0
   179
	case ENoFlush:
sl@0
   180
			err = deflate_r(&iStream,Z_NO_FLUSH);
sl@0
   181
		
sl@0
   182
			switch (err)
sl@0
   183
				{
sl@0
   184
			case Z_STREAM_ERROR:
sl@0
   185
				User::Leave(KEZlibErrStream);
sl@0
   186
				break;
sl@0
   187
sl@0
   188
			case Z_OK:
sl@0
   189
				if (iStream.avail_in == 0)
sl@0
   190
					iBufferInit->NeedInputL(*this);
sl@0
   191
				
sl@0
   192
				if (iStream.avail_out == 0)
sl@0
   193
					iBufferInit->NeedOutputL(*this);
sl@0
   194
				break;
sl@0
   195
sl@0
   196
			case Z_BUF_ERROR:  
sl@0
   197
				// this is probably ok we have just run out of input.
sl@0
   198
sl@0
   199
				iDeflationState = EFinish;
sl@0
   200
				break;
sl@0
   201
sl@0
   202
			default:
sl@0
   203
sl@0
   204
				// there's something wrong with this code if we get here !
sl@0
   205
				
sl@0
   206
				User::Leave(KEZlibErrUnexpected);
sl@0
   207
				break;
sl@0
   208
				}
sl@0
   209
sl@0
   210
			break;
sl@0
   211
sl@0
   212
	case EFinish:
sl@0
   213
		err = deflate_r(&iStream,Z_FINISH);
sl@0
   214
		
sl@0
   215
		switch (err)
sl@0
   216
			{
sl@0
   217
		case Z_STREAM_ERROR:				
sl@0
   218
				User::Leave(KEZlibErrStream);
sl@0
   219
				break;
sl@0
   220
sl@0
   221
		case Z_BUF_ERROR:
sl@0
   222
				User::Leave(KEZlibErrBuf);
sl@0
   223
				break;
sl@0
   224
sl@0
   225
		case Z_OK:
sl@0
   226
				if (iStream.avail_in == 0)
sl@0
   227
					iBufferInit->NeedInputL(*this);
sl@0
   228
				
sl@0
   229
				if (iStream.avail_out == 0)
sl@0
   230
					iBufferInit->NeedOutputL(*this);
sl@0
   231
				break;
sl@0
   232
sl@0
   233
		case Z_STREAM_END:
sl@0
   234
				iDeflationState = EFinalize;
sl@0
   235
				break;
sl@0
   236
			
sl@0
   237
		default:
sl@0
   238
				// there's something wrong with this code if we get here !
sl@0
   239
sl@0
   240
				User::Leave(KEZlibErrUnexpected);
sl@0
   241
				break;
sl@0
   242
			}
sl@0
   243
sl@0
   244
		break;
sl@0
   245
sl@0
   246
	case EFinalize:
sl@0
   247
		iBufferInit->FinalizeL(*this);
sl@0
   248
		callAgain = EFalse;
sl@0
   249
		iDeflationState = ETerminated;
sl@0
   250
		break;
sl@0
   251
sl@0
   252
	case ETerminated:
sl@0
   253
		User::Leave(KEZlibErrDeflateTerminated);
sl@0
   254
		}
sl@0
   255
sl@0
   256
	return callAgain;
sl@0
   257
	}
sl@0
   258
sl@0
   259
void CEZCompressor::ConstructL(TInt aLevel, const TUint8 *aDictionary, TInt aLength, 
sl@0
   260
						   TInt aWindowBits, TInt aMemLevel, TStrategy aStrategy)
sl@0
   261
	{
sl@0
   262
	ConstructL(aLevel,aWindowBits,aMemLevel,aStrategy);
sl@0
   263
	if (deflateSetDictionary_r(&iStream,STATIC_CAST(const Bytef *,aDictionary),aLength) == 
sl@0
   264
			Z_STREAM_ERROR)
sl@0
   265
		User::Leave(KEZlibErrStream);  // This should never happen.
sl@0
   266
	}
sl@0
   267
	
sl@0
   268
void CEZCompressor::ConstructL(TInt aLevel, TInt aWindowBits, TInt aMemLevel, TStrategy aStrategy)
sl@0
   269
	{
sl@0
   270
	// don't need to assert the validity of aWindowBits, aMemLevel & aStrategy as deflateInit2 will
sl@0
   271
	// do this for us.
sl@0
   272
sl@0
   273
	iStream.zalloc = Z_NULL;
sl@0
   274
	iStream.zfree = Z_NULL;
sl@0
   275
	iStream.opaque = Z_NULL;
sl@0
   276
sl@0
   277
	iBufferInit->InitializeL(*this);
sl@0
   278
sl@0
   279
	TInt err = deflateInit2_r(&iStream,aLevel,Z_DEFLATED,aWindowBits,aMemLevel, STATIC_CAST(int,aStrategy));
sl@0
   280
	if (err == Z_STREAM_ERROR)
sl@0
   281
		User::Leave(KEZlibErrStream);
sl@0
   282
	else if (err == Z_MEM_ERROR)
sl@0
   283
		User::LeaveNoMemory();
sl@0
   284
sl@0
   285
	iDeflationState = ENoFlush;
sl@0
   286
	}
sl@0
   287
sl@0
   288
/**
sl@0
   289
Compresses the data in the given buffer
sl@0
   290
sl@0
   291
@param aDestination the target buffer for the compressed data
sl@0
   292
@param aSource the buffer containing the data to be compressed
sl@0
   293
@param aLevel the level of compression
sl@0
   294
@leave KEZLibErrBuf There is a problem with the buffer
sl@0
   295
@leave KEZLIbErrStream There is a problem with the stream
sl@0
   296
@leave ... Any of the system wide error codes
sl@0
   297
*/
sl@0
   298
EXPORT_C void CEZCompressor::CompressL(TDes8 &aDestination, const TDesC8 &aSource, 
sl@0
   299
										TInt aLevel)
sl@0
   300
	{
sl@0
   301
	Bytef* destinationBuffer = STATIC_CAST(Bytef* ,CONST_CAST(TUint8* ,aDestination.Ptr()));
sl@0
   302
	const Bytef* sourceBuffer = STATIC_CAST(const Bytef* ,aSource.Ptr());
sl@0
   303
	uLongf dl = aDestination.MaxSize();
sl@0
   304
	TInt err = compress2_r(destinationBuffer,&dl,sourceBuffer,aSource.Size(),aLevel);
sl@0
   305
sl@0
   306
	if (err == Z_MEM_ERROR) 
sl@0
   307
		User::LeaveNoMemory();
sl@0
   308
	else if (err == Z_BUF_ERROR)
sl@0
   309
		User::Leave(KEZlibErrBuf);
sl@0
   310
	else if (err == Z_STREAM_ERROR)
sl@0
   311
		User::Leave(KEZlibErrStream);
sl@0
   312
sl@0
   313
	aDestination.SetLength(dl);
sl@0
   314
	}