crypto/weakcrypto/test/tpkcs5kdf/tactionderivekey.cpp
author hgs
Thu, 24 Jun 2010 15:39:07 +0530
changeset 72 de46a57f75fb
permissions -rw-r--r--
201023_02
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
72
de46a57f75fb 201023_02
hgs
parents:
diff changeset
     1
/*
de46a57f75fb 201023_02
hgs
parents:
diff changeset
     2
* Copyright (c) 1998-2009 Nokia Corporation and/or its subsidiary(-ies).
de46a57f75fb 201023_02
hgs
parents:
diff changeset
     3
* All rights reserved.
de46a57f75fb 201023_02
hgs
parents:
diff changeset
     4
* This component and the accompanying materials are made available
de46a57f75fb 201023_02
hgs
parents:
diff changeset
     5
* under the terms of the License "Eclipse Public License v1.0"
de46a57f75fb 201023_02
hgs
parents:
diff changeset
     6
* which accompanies this distribution, and is available
de46a57f75fb 201023_02
hgs
parents:
diff changeset
     7
* at the URL "http://www.eclipse.org/legal/epl-v10.html".
de46a57f75fb 201023_02
hgs
parents:
diff changeset
     8
*
de46a57f75fb 201023_02
hgs
parents:
diff changeset
     9
* Initial Contributors:
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    10
* Nokia Corporation - initial contribution.
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    11
*
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    12
* Contributors:
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    13
*
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    14
* Description: 
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    15
*
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    16
*/
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    17
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    18
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    19
#include "tactionderivekey.h"
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    20
#include "t_input.h"
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    21
#include <pkcs5kdf.h>
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    22
#include <pkcs12kdf.h>
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    23
#include <stdlib.h>
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    24
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    25
_LIT8(KKdfStart, "<kdf>");
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    26
_LIT8(KKdfEnd, "</kdf>");
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    27
_LIT8(KPKCS12Kdf, "pkcs#12");
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    28
_LIT8(KDeriveKeyStart, "<derivekey>");
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    29
_LIT8(KDeriveKeyEnd, "</derivekey>");
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    30
_LIT8(KKeyStart, "<key>");
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    31
_LIT8(KKeyEnd, "</key>");
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    32
_LIT8(KPasswdStart, "<passwd>");
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    33
_LIT8(KPasswdEnd, "</passwd>");
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    34
_LIT8(KSaltStart, "<salt>");
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    35
_LIT8(KSaltEnd, "</salt>");
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    36
_LIT8(KIterationsStart, "<iterations>");
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    37
_LIT8(KIterationsEnd, "</iterations>");
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    38
_LIT8(KLeaveInPerformAction, "<leaveinperformaction>");
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    39
_LIT8(KLeaveInPerformActionEnd, "</leaveinperformaction>");
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    40
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    41
CTestAction* CActionDeriveKey::NewL(RFs& aFs,
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    42
									   CConsoleBase& aConsole,
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    43
									   Output& aOut, 
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    44
									   const TTestActionSpec& aTestActionSpec)
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    45
	{
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    46
	CTestAction* self = CActionDeriveKey::NewLC(aFs, aConsole,
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    47
		aOut, aTestActionSpec);
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    48
	CleanupStack::Pop();
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    49
	return self;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    50
	}
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    51
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    52
CTestAction* CActionDeriveKey::NewLC(RFs& aFs,
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    53
										CConsoleBase& aConsole,
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    54
										Output& aOut, 
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    55
										const TTestActionSpec& aTestActionSpec)
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    56
	{
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    57
	CActionDeriveKey* self = new(ELeave) CActionDeriveKey(aFs, aConsole, aOut);
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    58
	CleanupStack::PushL(self);
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    59
	self->ConstructL(aTestActionSpec);
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    60
	return self;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    61
	}
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    62
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    63
CActionDeriveKey::~CActionDeriveKey()
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    64
	{
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    65
	delete iBody;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    66
	}
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    67
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    68
CActionDeriveKey::CActionDeriveKey(RFs& aFs, 
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    69
								 CConsoleBase& aConsole,
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    70
								 Output& aOut)
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    71
								 
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    72
: CTestAction(aConsole, aOut), iFs(aFs)
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    73
	{
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    74
	}
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    75
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    76
void CActionDeriveKey::ConstructL(const TTestActionSpec& aTestActionSpec)
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    77
	{
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    78
	CTestAction::ConstructL(aTestActionSpec);
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    79
	iBody = HBufC8::NewL(aTestActionSpec.iActionBody.Length());
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    80
	iBody->Des().Copy(aTestActionSpec.iActionBody);
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    81
	}
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    82
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    83
void CActionDeriveKey::DoPerformPrerequisite(TRequestStatus& aStatus)
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    84
	{
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    85
	TRequestStatus* status = &aStatus;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    86
	TInt err = KErrNone;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    87
	TInt pos = 0;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    88
	TPtrC8 deriveKey = Input::ParseElement(*iBody, KDeriveKeyStart, KDeriveKeyEnd, pos, err);
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    89
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    90
	// KDF is only explicitly specified for PKCS#12 derived keys
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    91
	pos = 0;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    92
	TPtrC8 kdfTemp = Input::ParseElement(deriveKey, KKdfStart, KKdfEnd, pos, err);
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    93
	if (err == KErrNone)
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    94
		iKdf = kdfTemp.AllocL();
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    95
	
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    96
	pos = 0;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    97
	TPtrC8 passwdTemp = Input::ParseElement(deriveKey, KPasswdStart, KPasswdEnd, pos, err);
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    98
	iPasswd = HBufC8::NewL(passwdTemp.Length());
de46a57f75fb 201023_02
hgs
parents:
diff changeset
    99
	*iPasswd = passwdTemp;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   100
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   101
	pos = 0;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   102
	TPtrC8 iterationsTemp = Input::ParseElement(deriveKey, KIterationsStart, KIterationsEnd, pos, err);
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   103
	iIterations = HBufC8::NewL(iterationsTemp.Length() + 1); //added 1 for the null zero used later
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   104
	*iIterations = iterationsTemp;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   105
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   106
	pos = 0;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   107
	TPtrC8 saltTemp = Input::ParseElement(deriveKey, KSaltStart, KSaltEnd, pos, err);
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   108
	iSalt = HBufC8::NewL(saltTemp.Length());
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   109
	*iSalt = saltTemp;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   110
	Hex(*iSalt);
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   111
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   112
	pos = 0;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   113
	TPtrC8 keyTemp = Input::ParseElement(deriveKey, KKeyStart, KKeyEnd, pos, err);
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   114
	iKey = HBufC8::NewL(keyTemp.Length());
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   115
	*iKey = keyTemp;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   116
	Hex(*iKey);
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   117
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   118
	iOutput = HBufC8::NewL(iKey->Length());
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   119
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   120
	pos = 0;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   121
	iLeaveInPerformAction = Input::ParseIntElement(deriveKey, 
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   122
		KLeaveInPerformAction, KLeaveInPerformActionEnd, pos, err);
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   123
	if (err)
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   124
		{
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   125
		iLeaveInPerformAction = 0;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   126
		}
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   127
	User::RequestComplete(status, KErrNone);
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   128
	iActionState = CTestAction::EAction;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   129
	}
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   130
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   131
void CActionDeriveKey::DoPerformPostrequisite(TRequestStatus& aStatus)
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   132
	{
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   133
	TRequestStatus* status = &aStatus;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   134
	delete iKey;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   135
	delete iSalt;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   136
	delete iIterations;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   137
	delete iPasswd;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   138
	delete iOutput;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   139
	delete iKdf;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   140
	iKdf = 0;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   141
	
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   142
	iFinished = ETrue;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   143
	User::RequestComplete(status, KErrNone);
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   144
	}
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   145
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   146
void CActionDeriveKey::DoReportAction(void)
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   147
	{
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   148
	}
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   149
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   150
void CActionDeriveKey::DoCheckResult(TInt)
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   151
	{
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   152
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   153
	}
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   154
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   155
void CActionDeriveKey::PerformAction(TRequestStatus& aStatus)
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   156
	{
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   157
	TRequestStatus* status = &aStatus;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   158
	iResult = EFalse;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   159
	
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   160
	if (iLeaveInPerformAction)
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   161
		{
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   162
		User::Leave(KErrArgument);
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   163
		}
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   164
	iOutput->Des().SetLength(iKey->Length());
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   165
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   166
	TUint8* nptr= (TUint8*)(iIterations->Des().PtrZ()); 
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   167
	TUint32 i = strtoul((char*)nptr, 0, 10); 
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   168
	
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   169
	iConsole.Printf(_L("."));
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   170
	TPtr8 outputActual = iOutput->Des();
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   171
	TPtr8 passwdActual = iPasswd->Des();
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   172
	if (iKdf != 0 && *iKdf == KPKCS12Kdf)
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   173
		{
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   174
		// convert the password to PKCS#12 password format
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   175
		HBufC* pwdNative = HBufC::NewLC(iPasswd->Length());
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   176
		pwdNative->Des().Copy(*iPasswd);
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   177
		HBufC8* pwdPKCS12 = PKCS12KDF::GeneratePasswordLC(*pwdNative);
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   178
		PKCS12KDF::DeriveKeyL(outputActual, PKCS12KDF::EIDByteEncryptKey, *pwdPKCS12, *iSalt, i);
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   179
		CleanupStack::PopAndDestroy(2, pwdNative);
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   180
		}
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   181
	else	// PKCS#5
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   182
		{
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   183
		TPtr8 saltActual = iSalt->Des();
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   184
		TPKCS5KDF::DeriveKeyL(outputActual, passwdActual, saltActual,i);
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   185
		}
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   186
		
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   187
	if(*iOutput == *iKey)
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   188
		{
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   189
		iResult = ETrue;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   190
		}
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   191
		
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   192
	User::RequestComplete(status, KErrNone);
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   193
	iActionState = CTestAction::EPostrequisite;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   194
	}
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   195
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   196
void CActionDeriveKey::Hex(HBufC8& aString)
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   197
/**
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   198
	Convert the supplied hex string into the binary equivalent.
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   199
	
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   200
	@param	aString			Hex string.  On entry this contains
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   201
							a sequence of hexadecimal characters,
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   202
							e.g., "3037AFC8EA".  On exit it is
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   203
							half the original length and each two-digit
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   204
							hex number is reduced to the matching
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   205
							byte value.
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   206
 */
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   207
    {
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   208
    TPtr8 ptr=aString.Des();
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   209
    if (aString.Length()%2)
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   210
        {
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   211
        ptr.SetLength(0);
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   212
        return;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   213
        }
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   214
    TInt i;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   215
    for (i=0;i<aString.Length();i+=2)
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   216
        {
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   217
        TUint8 tmp;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   218
        tmp=(TUint8)(aString[i]-(aString[i]>'9'?('A'-10):'0'));
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   219
        tmp*=16;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   220
        tmp|=(TUint8)(aString[i+1]-(aString[i+1]>'9'?('A'-10):'0'));
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   221
        ptr[i/2]=tmp;
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   222
        }
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   223
    ptr.SetLength(aString.Length()/2);
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   224
    }
de46a57f75fb 201023_02
hgs
parents:
diff changeset
   225