os/ossrv/stdlibs/libcrypt/src/encrypt.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
/*
sl@0
     2
* Copyright (c) 2005-2006 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     3
* All rights reserved.
sl@0
     4
* This component and the accompanying materials are made available
sl@0
     5
* under the terms of "Eclipse Public License v1.0"
sl@0
     6
* which accompanies this distribution, and is available
sl@0
     7
* at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     8
*
sl@0
     9
* Initial Contributors:
sl@0
    10
* Nokia Corporation - initial contribution.
sl@0
    11
*
sl@0
    12
* Contributors:
sl@0
    13
*
sl@0
    14
* Description:  Contains implementation for encryption/decryption.
sl@0
    15
*
sl@0
    16
*/
sl@0
    17
sl@0
    18
sl@0
    19
// INCLUDE FILES
sl@0
    20
sl@0
    21
#define EMULATOR ((defined(__WINS__) || defined(__WINSCW__)))
sl@0
    22
sl@0
    23
#include <e32def.h>
sl@0
    24
//#include <des.h>
sl@0
    25
#include <hash.h>
sl@0
    26
#include <errno.h>
sl@0
    27
#include <string.h>
sl@0
    28
sl@0
    29
// EXTERNAL FUNCTION PROTOTYPES
sl@0
    30
extern "C" char *crypt_des(const char *key, const char *setting);
sl@0
    31
extern "C" char *crypt_md5(const char *pw, const char *salt);
sl@0
    32
sl@0
    33
// LOCAL CONSTANTS AND MACROS
sl@0
    34
#define BYTE_SIZE	8
sl@0
    35
#define ENCRYPTION  0
sl@0
    36
#define DECRYPTION  1
sl@0
    37
sl@0
    38
// STATIC DATA
sl@0
    39
#if !EMULATOR
sl@0
    40
TBuf8<BYTE_SIZE> desKey;				// For persistence between calls 
sl@0
    41
										// to setkey() and encrypt()
sl@0
    42
static TInt bSetkeyInvoked = 0;
sl@0
    43
#else
sl@0
    44
#include <sys/types.h>
sl@0
    45
#include "wsd_solution.h"
sl@0
    46
#define bSetkeyInvoked (GetGlobals()->bSetkeyInvoked)
sl@0
    47
#endif
sl@0
    48
sl@0
    49
// LOCAL FUNCTION PROTOTYPES
sl@0
    50
static unsigned char GetByte(const char *bitVector);
sl@0
    51
static void DesEncryptionL(const TDes8& aKey, TDes8& aInputBlock);
sl@0
    52
static void DesDecryptionL(const TDes8& aKey, TDes8& aInputBlock);
sl@0
    53
sl@0
    54
// LOCAL class declaration
sl@0
    55
class CEncDecHack : public CBase
sl@0
    56
{
sl@0
    57
public:
sl@0
    58
	virtual void Transform(TDes8& aBlock){}
sl@0
    59
};
sl@0
    60
sl@0
    61
typedef CEncDecHack* (*LookupFuncEncDecObjCreator)(const TDesC8& aKey, TBool aCheckWeakKey);
sl@0
    62
sl@0
    63
_LIT(KCryptoDll,"cryptography.dll");
sl@0
    64
sl@0
    65
// -----------------------------------------------------------------------------
sl@0
    66
// function_name: _setkey
sl@0
    67
//
sl@0
    68
// Prepares a byte array for the key from the contents of the incoming bit vector.
sl@0
    69
// Key thus constructed is statically stored for use during encryption/decryption.
sl@0
    70
//
sl@0
    71
// Returns: void
sl@0
    72
// -----------------------------------------------------------------------------
sl@0
    73
//
sl@0
    74
extern "C"
sl@0
    75
void _setkey (const char *key)
sl@0
    76
{
sl@0
    77
#if !EMULATOR
sl@0
    78
    // Reset the contents of the 'key' descriptor
sl@0
    79
    desKey.Delete(0,desKey.Length());
sl@0
    80
#endif
sl@0
    81
sl@0
    82
#if !EMULATOR    
sl@0
    83
	// Pack the contents of the bit vector into a TDes derived object
sl@0
    84
	for( int i = 0 ; i < BYTE_SIZE ; ++i)
sl@0
    85
	{
sl@0
    86
		desKey.Append( GetByte( &key[i * BYTE_SIZE] ) );
sl@0
    87
	}
sl@0
    88
#else
sl@0
    89
	for(int i=0 ; i<64 ; ++i)
sl@0
    90
	{
sl@0
    91
		(GetGlobals()->desKey)[i] = key[i];
sl@0
    92
	}
sl@0
    93
#endif
sl@0
    94
sl@0
    95
	bSetkeyInvoked = 1;
sl@0
    96
}
sl@0
    97
sl@0
    98
// -----------------------------------------------------------------------------
sl@0
    99
// function_name: _encrypt
sl@0
   100
//
sl@0
   101
// Performs either encryption or decryption of the data block. Prior to invoking
sl@0
   102
// Symbian OS cryptography APIs for encryptions/decryption, this function
sl@0
   103
// packs the contents of the bit vector into a byte array of size eight. The byte
sl@0
   104
// array obtained after encryption/decryption is unpacked to present the output
sl@0
   105
// in the form of a bit vector. The incoming data block is modified in place 
sl@0
   106
// during the process.
sl@0
   107
//
sl@0
   108
// Assumption: User of the libcrypt library is expected the create a cleanupstack
sl@0
   109
//
sl@0
   110
// Returns: void
sl@0
   111
// -----------------------------------------------------------------------------
sl@0
   112
//
sl@0
   113
extern "C"
sl@0
   114
void _encrypt (char block[], int edflag)
sl@0
   115
{
sl@0
   116
#if EMULATOR
sl@0
   117
	TBuf8<BYTE_SIZE> desKey;
sl@0
   118
#endif
sl@0
   119
sl@0
   120
	// Determine if setkey() is invoked by the user
sl@0
   121
	if(!bSetkeyInvoked)
sl@0
   122
	{
sl@0
   123
		// Initialize the key with default values
sl@0
   124
		for(int i = 0 ; i < BYTE_SIZE ; ++i)
sl@0
   125
		{
sl@0
   126
			desKey.Append((unsigned char)0);
sl@0
   127
		}
sl@0
   128
		bSetkeyInvoked = 1;
sl@0
   129
	}
sl@0
   130
#if EMULATOR
sl@0
   131
	else
sl@0
   132
	{
sl@0
   133
		for(int i=0 ; i<BYTE_SIZE ; ++i)
sl@0
   134
		{
sl@0
   135
			desKey.Append( GetByte( (const char*)&(GetGlobals()->desKey)[i * BYTE_SIZE] ));
sl@0
   136
		}
sl@0
   137
	}
sl@0
   138
#endif
sl@0
   139
sl@0
   140
	// Determine whether encryption or decryption is requested
sl@0
   141
	if(edflag != ENCRYPTION)
sl@0
   142
	{
sl@0
   143
		if(edflag != DECRYPTION)
sl@0
   144
		{
sl@0
   145
			// Unrecognized flag parameter
sl@0
   146
			errno = EPERM;
sl@0
   147
			return;
sl@0
   148
		}
sl@0
   149
	}
sl@0
   150
	
sl@0
   151
	// Pack the contents of the input bit vector into a "byte" array
sl@0
   152
	TBuf8<BYTE_SIZE> inputBlock;
sl@0
   153
	TInt nIterator;
sl@0
   154
	for(nIterator = 0 ; nIterator < BYTE_SIZE ; ++nIterator)
sl@0
   155
	{
sl@0
   156
		inputBlock.Append( GetByte( &block[nIterator * BYTE_SIZE] ) );
sl@0
   157
	}
sl@0
   158
	
sl@0
   159
	TInt error = KErrNone;
sl@0
   160
	typedef void (*DesOperation)(const TDes8&, TDes8&);
sl@0
   161
	DesOperation funcOperationL = NULL;
sl@0
   162
sl@0
   163
	switch(edflag)
sl@0
   164
	{
sl@0
   165
		case ENCRYPTION:		  // Encryption
sl@0
   166
		funcOperationL = DesEncryptionL;
sl@0
   167
		break;
sl@0
   168
sl@0
   169
		case DECRYPTION:		  // Decryption
sl@0
   170
		funcOperationL = DesDecryptionL;
sl@0
   171
		break;
sl@0
   172
	}
sl@0
   173
sl@0
   174
	TRAP(error, (*funcOperationL)(desKey, inputBlock));
sl@0
   175
	
sl@0
   176
	if(error == KErrNone)
sl@0
   177
	{
sl@0
   178
		unsigned char chTemp;
sl@0
   179
		int k = 0;
sl@0
   180
sl@0
   181
		// Create the bit vector from the "byte" array (unpack)
sl@0
   182
		for(int i = 0 ; i < BYTE_SIZE ; ++i)
sl@0
   183
		{
sl@0
   184
			chTemp = inputBlock[i];
sl@0
   185
			for(int j = 0 ; j < BYTE_SIZE ; ++j)
sl@0
   186
			{
sl@0
   187
				block[k++] = ((chTemp & 0x80) >> 7);
sl@0
   188
				chTemp <<= 1;
sl@0
   189
			}
sl@0
   190
		}
sl@0
   191
	}
sl@0
   192
	else
sl@0
   193
	{
sl@0
   194
		// Set the errno flag to indicate failure
sl@0
   195
		errno = EPERM;
sl@0
   196
	}
sl@0
   197
}
sl@0
   198
sl@0
   199
// -----------------------------------------------------------------------------
sl@0
   200
// function_name: _crypt
sl@0
   201
//
sl@0
   202
// Uses MD5-based algorithm or DES encryption mechanism to encode a constant
sl@0
   203
// string using "key" as the key. Salt determines the algorithm to be used.
sl@0
   204
//
sl@0
   205
// Returns: pointer to a static data buffer containing the encoded "string"
sl@0
   206
// -----------------------------------------------------------------------------
sl@0
   207
//
sl@0
   208
extern "C" 
sl@0
   209
char* _crypt (const char *key, const char *salt)
sl@0
   210
{
sl@0
   211
	// Identify the algorithm to be used as part of crypt
sl@0
   212
	if(strstr(salt, "$1$"))
sl@0
   213
	{
sl@0
   214
		// MD5-based algorithm
sl@0
   215
		return crypt_md5(key, salt);
sl@0
   216
	}
sl@0
   217
	else
sl@0
   218
	{
sl@0
   219
		return crypt_des(key, salt);
sl@0
   220
	}
sl@0
   221
}
sl@0
   222
sl@0
   223
// -----------------------------------------------------------------------------
sl@0
   224
// function_name: GetByte
sl@0
   225
//
sl@0
   226
// Packs the "bits" in the bit vector into a byte
sl@0
   227
//
sl@0
   228
// Returns: Byte composed of the bits from the bit vector
sl@0
   229
// -----------------------------------------------------------------------------
sl@0
   230
//
sl@0
   231
LOCAL_C unsigned char GetByte(const char *bitVector)
sl@0
   232
{
sl@0
   233
	unsigned char chTemp = 0;
sl@0
   234
sl@0
   235
	for(int nIterator = 0 ; nIterator < BYTE_SIZE ; ++nIterator)
sl@0
   236
	{
sl@0
   237
		chTemp |= ( bitVector[nIterator] << (BYTE_SIZE - nIterator - 1) );
sl@0
   238
	}
sl@0
   239
	return chTemp;
sl@0
   240
}
sl@0
   241
sl@0
   242
// -----------------------------------------------------------------------------
sl@0
   243
// function_name: DesEncryptionL
sl@0
   244
//
sl@0
   245
// Function to encrypt the input data bytes by invoking Symbian OS API for
sl@0
   246
// DES algorithm for encryption
sl@0
   247
//
sl@0
   248
// Assumption: 1. BLOCKSIZE within the cryptography library is 8 for 
sl@0
   249
// 				  DES encryption
sl@0
   250
//			   2. The input key is not checked against a set of known
sl@0
   251
//				  weak key values
sl@0
   252
//
sl@0
   253
// Returns: void, however, this function leaves if there is insufficient
sl@0
   254
//          memory
sl@0
   255
// -----------------------------------------------------------------------------
sl@0
   256
//
sl@0
   257
LOCAL_C void DesEncryptionL(const TDes8& aKey, TDes8& aInputBlock)
sl@0
   258
{
sl@0
   259
	// Construct the encryptor object
sl@0
   260
/*	CDESEncryptor *pEncryptor = CDESEncryptor::NewL(aKey, EFalse);
sl@0
   261
	
sl@0
   262
	if(!pEncryptor)
sl@0
   263
	{
sl@0
   264
		User::Leave(KErrNoMemory);
sl@0
   265
	}
sl@0
   266
	
sl@0
   267
	// Invoke DES trasnformation to encrypt the input data
sl@0
   268
	pEncryptor->Transform(aInputBlock);
sl@0
   269
	
sl@0
   270
	delete pEncryptor;
sl@0
   271
*/
sl@0
   272
	RLibrary library;
sl@0
   273
    User::LeaveIfError(library.Load(KCryptoDll));
sl@0
   274
    
sl@0
   275
    #ifdef __WINSCW__
sl@0
   276
   	TLibraryFunction func = library.Lookup(102); // CDESEncryptor::NewL
sl@0
   277
    #else
sl@0
   278
    TLibraryFunction func = library.Lookup(59);  //CDESEncryptor::NewL
sl@0
   279
	#endif // ifdef __WINSCW__
sl@0
   280
	
sl@0
   281
    if (func == NULL)
sl@0
   282
    {
sl@0
   283
        library.Close();
sl@0
   284
        User::Leave(KErrNotFound);
sl@0
   285
    }
sl@0
   286
    LookupFuncEncDecObjCreator  objCreatorFuncion = reinterpret_cast<LookupFuncEncDecObjCreator> (func);
sl@0
   287
    CEncDecHack* pEncryptor = reinterpret_cast<CEncDecHack*>(objCreatorFuncion(aKey, EFalse));
sl@0
   288
    pEncryptor->Transform(aInputBlock);
sl@0
   289
    delete pEncryptor;
sl@0
   290
    library.Close();
sl@0
   291
}
sl@0
   292
sl@0
   293
// -----------------------------------------------------------------------------
sl@0
   294
// function_name: DesDecryptionL
sl@0
   295
//
sl@0
   296
// Function to encrypt the input data bytes by invoking Symbian OS API for
sl@0
   297
// DES algorithm for decryption
sl@0
   298
//
sl@0
   299
// Assumption: 1. BLOCKSIZE within the cryptography library is 8 for 
sl@0
   300
// 				  DES decryption
sl@0
   301
//			   2. The input key is not checked against a set of known
sl@0
   302
//				  weak key values
sl@0
   303
//
sl@0
   304
// Returns: void, however, this function leaves if there is insufficient
sl@0
   305
//          memory
sl@0
   306
// -----------------------------------------------------------------------------
sl@0
   307
//
sl@0
   308
LOCAL_C void DesDecryptionL(const TDes8& aKey, TDes8& aInputBlock)
sl@0
   309
{
sl@0
   310
	// Construct the decryptor object
sl@0
   311
/*	CDESDecryptor *pDecryptor = CDESDecryptor::NewL(aKey, EFalse);
sl@0
   312
	
sl@0
   313
	if(!pDecryptor)
sl@0
   314
	{
sl@0
   315
		User::Leave(KErrNoMemory);
sl@0
   316
	}
sl@0
   317
	
sl@0
   318
	// Invoke DES decryption on the cipher text
sl@0
   319
	pDecryptor->Transform(aInputBlock);
sl@0
   320
	
sl@0
   321
	delete pDecryptor;
sl@0
   322
*/
sl@0
   323
    RLibrary library;
sl@0
   324
    User::LeaveIfError(library.Load(KCryptoDll));
sl@0
   325
    
sl@0
   326
	#ifdef __WINSCW__
sl@0
   327
    TLibraryFunction func = library.Lookup(101); // CDESDecryptor::NewL
sl@0
   328
	#else
sl@0
   329
    TLibraryFunction func = library.Lookup(57);  //CDESDecryptor::NewL
sl@0
   330
	#endif // ifdef __WINSCW
sl@0
   331
	
sl@0
   332
    if (func == NULL)
sl@0
   333
    {
sl@0
   334
        library.Close();
sl@0
   335
        return;
sl@0
   336
    }
sl@0
   337
    LookupFuncEncDecObjCreator  objCreatorFuncion = reinterpret_cast<LookupFuncEncDecObjCreator> (func);
sl@0
   338
    CEncDecHack* pDecryptor = reinterpret_cast<CEncDecHack*>(objCreatorFuncion(aKey, EFalse));
sl@0
   339
    pDecryptor->Transform(aInputBlock);
sl@0
   340
    delete pDecryptor;
sl@0
   341
    library.Close();
sl@0
   342
}
sl@0
   343
sl@0
   344
extern "C" {
sl@0
   345
sl@0
   346
// -----------------------------------------------------------------------------
sl@0
   347
// function_name: Deallocate2DimensionalUchar
sl@0
   348
//
sl@0
   349
// To deallocate storage alloted for the two dimensional array
sl@0
   350
//
sl@0
   351
// Returns: void
sl@0
   352
// -----------------------------------------------------------------------------
sl@0
   353
//
sl@0
   354
void Deallocate2DimensionalUchar(unsigned char **buffer, int row)
sl@0
   355
{
sl@0
   356
		int m;
sl@0
   357
		for(m=0 ; m<row ; ++m)
sl@0
   358
		{
sl@0
   359
			User::Free((TAny *)buffer[m]);
sl@0
   360
		}
sl@0
   361
		User::Free((TAny *)buffer);
sl@0
   362
}
sl@0
   363
	
sl@0
   364
// -----------------------------------------------------------------------------
sl@0
   365
// function_name: Deallocate2DimensionalUint
sl@0
   366
//
sl@0
   367
// To deallocate storage alloted for the two dimensional array
sl@0
   368
//
sl@0
   369
// Returns: void
sl@0
   370
// -----------------------------------------------------------------------------
sl@0
   371
//
sl@0
   372
void Deallocate2DimensionalUint(__uint32_t **buffer, int row)
sl@0
   373
{
sl@0
   374
		int m;
sl@0
   375
		for(m=0 ; m<row ; ++m)
sl@0
   376
		{
sl@0
   377
			User::Free((TAny *)buffer[m]);
sl@0
   378
		}
sl@0
   379
		User::Free((TAny *)buffer);
sl@0
   380
}
sl@0
   381
sl@0
   382
// -----------------------------------------------------------------------------
sl@0
   383
// function_name: Allocate2DimensionalUchar
sl@0
   384
//
sl@0
   385
// Function to allocate storage for a two dimensional array from the heap. 
sl@0
   386
// If heap exhaustion occurs during the course of allocating memory to the array,
sl@0
   387
// the previously allocated storage will be deleted prior to returning to the
sl@0
   388
// caller.
sl@0
   389
//
sl@0
   390
// Returns: non-zero if allocation is successful, 0 if it fails
sl@0
   391
// -----------------------------------------------------------------------------
sl@0
   392
//
sl@0
   393
int Allocate2DimensionalUchar(unsigned char ***buffer, int row, int column)
sl@0
   394
{
sl@0
   395
	*buffer = (unsigned char **)User::Alloc(row * sizeof(unsigned char *));
sl@0
   396
	if(NULL == *buffer)
sl@0
   397
	{
sl@0
   398
		return 0;
sl@0
   399
	}
sl@0
   400
	for(int m=0 ; m<row ; ++m)
sl@0
   401
	{
sl@0
   402
		(*buffer)[m] = (unsigned char *)User::Alloc(column * sizeof(unsigned char));
sl@0
   403
		if(NULL == (*buffer)[m])
sl@0
   404
		{
sl@0
   405
			// Insufficient heap memory
sl@0
   406
			if(m)
sl@0
   407
			{
sl@0
   408
				// Deallocate the previously allocated storage
sl@0
   409
				Deallocate2DimensionalUchar(*buffer, m);
sl@0
   410
			}
sl@0
   411
			return 0;
sl@0
   412
		}
sl@0
   413
	}
sl@0
   414
	
sl@0
   415
	return 1;
sl@0
   416
}
sl@0
   417
sl@0
   418
// -----------------------------------------------------------------------------
sl@0
   419
// function_name: Allocate2DimensionalUint
sl@0
   420
//
sl@0
   421
// Function to allocate storage for a two dimensional array of type unsigned int
sl@0
   422
// from the heap. If heap exhaustion occurs while allocating memory to the array,
sl@0
   423
// the previously allocated storage will be deleted prior to returning to the
sl@0
   424
// caller.
sl@0
   425
//
sl@0
   426
// Returns: non-zero if allocation is successful, 0 if it fails
sl@0
   427
// -----------------------------------------------------------------------------
sl@0
   428
//
sl@0
   429
int Allocate2DimensionalUint(__uint32_t ***buffer, int row, int column)
sl@0
   430
{
sl@0
   431
	*buffer = (__uint32_t **)User::Alloc(row * sizeof(__uint32_t *));
sl@0
   432
	if(NULL == *buffer)
sl@0
   433
	{
sl@0
   434
		return 0;
sl@0
   435
	}
sl@0
   436
	for(int m=0 ; m<row ; ++m)
sl@0
   437
	{
sl@0
   438
		(*buffer)[m] = (__uint32_t *)User::Alloc(column * sizeof(__uint32_t));
sl@0
   439
		if(NULL == (*buffer)[m])
sl@0
   440
		{
sl@0
   441
			// Insufficient heap memory
sl@0
   442
			if(m)
sl@0
   443
			{
sl@0
   444
				// Deallocate the previously allocated storage
sl@0
   445
				Deallocate2DimensionalUint(*buffer, m);
sl@0
   446
			}
sl@0
   447
			return 0;
sl@0
   448
		}
sl@0
   449
	}
sl@0
   450
	
sl@0
   451
	return 1;
sl@0
   452
}
sl@0
   453
sl@0
   454
}	// <<end extern "C">>