os/ossrv/ssl/libcrypto/src/crypto/certretriever/certretriever.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
/*
sl@0
     2
Copyright (c) 2009-2010 Nokia Corporation and/or its subsidiary(-ies). All rights reserved.
sl@0
     3
sl@0
     4
Redistribution and use in source and binary forms, with or without 
sl@0
     5
modification, are permitted provided that the following conditions are met:
sl@0
     6
sl@0
     7
* Redistributions of source code must retain the above copyright notice, this 
sl@0
     8
  list of conditions and the following disclaimer.
sl@0
     9
* Redistributions in binary form must reproduce the above copyright notice, 
sl@0
    10
  this list of conditions and the following disclaimer in the documentation 
sl@0
    11
  and/or other materials provided with the distribution.
sl@0
    12
* Neither the name of Nokia Corporation nor the names of its contributors 
sl@0
    13
  may be used to endorse or promote products derived from this software 
sl@0
    14
  without specific prior written permission.
sl@0
    15
sl@0
    16
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" 
sl@0
    17
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 
sl@0
    18
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE 
sl@0
    19
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE 
sl@0
    20
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 
sl@0
    21
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR 
sl@0
    22
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER 
sl@0
    23
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 
sl@0
    24
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 
sl@0
    25
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
sl@0
    26
sl@0
    27
Description:  Contains implementation for x509_add_symbian_cert - to use certificates installed in Symbian with OpenSSL code.
sl@0
    28
*/
sl@0
    29
sl@0
    30
 
sl@0
    31
#include "certretriever.h"
sl@0
    32
#include "createx509.h"
sl@0
    33
sl@0
    34
#include <securitydefsconst.h>
sl@0
    35
sl@0
    36
#ifdef __cplusplus
sl@0
    37
extern "C" 
sl@0
    38
{
sl@0
    39
#endif
sl@0
    40
int X509_add_symbian_certsL(X509_STORE * store)
sl@0
    41
{
sl@0
    42
	CActiveScheduler* activeScheduler;
sl@0
    43
	CActiveScheduler* CurrentActiveScheduler = CActiveScheduler::Current();
sl@0
    44
	if(CurrentActiveScheduler == NULL)
sl@0
    45
	{
sl@0
    46
		activeScheduler = new (ELeave) CActiveScheduler;
sl@0
    47
		CleanupStack::PushL(activeScheduler);
sl@0
    48
		
sl@0
    49
		CActiveScheduler::Install(activeScheduler);
sl@0
    50
	}
sl@0
    51
	
sl@0
    52
		
sl@0
    53
	TRequestStatus status;
sl@0
    54
	CCertRetriever* certRetriever;
sl@0
    55
	TRAPD(error, certRetriever = CCertRetriever::NewL(store, status, CActiveScheduler::Current()));
sl@0
    56
	
sl@0
    57
	// create CActiveSchedulerWait
sl@0
    58
	if (CurrentActiveScheduler)
sl@0
    59
	{
sl@0
    60
		certRetriever->activeSchedulerwait = new (ELeave) CActiveSchedulerWait; 
sl@0
    61
		certRetriever->OwnScheduler = EFalse;	
sl@0
    62
	}
sl@0
    63
	
sl@0
    64
	
sl@0
    65
	if(error != KErrNone)
sl@0
    66
	{
sl@0
    67
		CleanupStack::PopAndDestroy();	// activeScheduler
sl@0
    68
		return 0;
sl@0
    69
	}
sl@0
    70
	
sl@0
    71
	CleanupStack::PushL(certRetriever);	
sl@0
    72
	
sl@0
    73
	TRAP(error,certRetriever->RetriveCertificateL());
sl@0
    74
	if(error != KErrNone)
sl@0
    75
	{
sl@0
    76
		if(CurrentActiveScheduler == NULL)
sl@0
    77
		{
sl@0
    78
			CleanupStack::PopAndDestroy(2);	// activeScheduler, certRetriever
sl@0
    79
		}
sl@0
    80
		else
sl@0
    81
			CleanupStack::PopAndDestroy();	// certRetriever
sl@0
    82
		return 0;
sl@0
    83
	}
sl@0
    84
		
sl@0
    85
				
sl@0
    86
	if(CurrentActiveScheduler == NULL)
sl@0
    87
	{
sl@0
    88
		activeScheduler->Start();
sl@0
    89
		 CleanupStack::PopAndDestroy(2); // If you destroy the object it will not be there
sl@0
    90
										   // in the iActiveQ, and hence stray signal.
sl@0
    91
		
sl@0
    92
		//CleanupStack::Pop(2);	   // So just pop it. // activeScheduler, certRetriever						   
sl@0
    93
	}
sl@0
    94
	else
sl@0
    95
	{
sl@0
    96
	//	CurrentActiveScheduler->Start();// If you are using CActiveScheduler::Current();
sl@0
    97
										  // Why u want to start it again?	
sl@0
    98
        
sl@0
    99
        // CleanupStack::PopAndDestroy(); // If you destroy the object it will not be there
sl@0
   100
										  // in the iActiveQ, and hence stray signal.	
sl@0
   101
										  									   
sl@0
   102
		// should wait here untill it finish loading certificates, ths API is synchronous
sl@0
   103
		certRetriever->activeSchedulerwait->Start();
sl@0
   104
		   
sl@0
   105
		CleanupStack::Pop(); 		  // So just pop it. // certRetriever
sl@0
   106
	}
sl@0
   107
	
sl@0
   108
	
sl@0
   109
		
sl@0
   110
	if(status == KErrNone)
sl@0
   111
		return 1;
sl@0
   112
	else
sl@0
   113
		return 0;
sl@0
   114
}
sl@0
   115
#ifdef __cplusplus
sl@0
   116
}
sl@0
   117
#endif
sl@0
   118
sl@0
   119
CCertRetriever::CCertRetriever(X509_STORE* aStore, 
sl@0
   120
				TRequestStatus& aStatus,
sl@0
   121
				const CActiveScheduler* aActiveScheduler)
sl@0
   122
			: CActive(CActive::EPriorityHigh), 
sl@0
   123
			  iStore(aStore),
sl@0
   124
			  iFinStatus ( aStatus ),
sl@0
   125
			  iActiveScheduler(aActiveScheduler),
sl@0
   126
			  iCertPtr(0,0)			  
sl@0
   127
			  
sl@0
   128
	{
sl@0
   129
	
sl@0
   130
	OwnScheduler = ETrue;
sl@0
   131
	if(iActiveScheduler)
sl@0
   132
		iActiveScheduler->Add(this);
sl@0
   133
	}
sl@0
   134
sl@0
   135
CCertRetriever::~CCertRetriever()
sl@0
   136
	{
sl@0
   137
		Cancel();
sl@0
   138
 		delete iBuf;
sl@0
   139
 		delete iCertFilter;
sl@0
   140
 		
sl@0
   141
 	 	iCerts.Close();	
sl@0
   142
 		delete iCertStore;
sl@0
   143
 		iFs.Close();
sl@0
   144
 		if(!OwnScheduler)
sl@0
   145
 		  delete activeSchedulerwait;
sl@0
   146
 	}
sl@0
   147
	
sl@0
   148
CCertRetriever* CCertRetriever::NewLC(X509_STORE* aStore,
sl@0
   149
						TRequestStatus& aStatus, 
sl@0
   150
						const CActiveScheduler* aActiveScheduler)
sl@0
   151
	{
sl@0
   152
	CCertRetriever* self = new (ELeave) CCertRetriever(aStore, aStatus, aActiveScheduler);
sl@0
   153
	CleanupStack::PushL(self);
sl@0
   154
	TRAPD(err,self->ConstructL());
sl@0
   155
	if(err != KErrNotFound)
sl@0
   156
		return self;
sl@0
   157
	
sl@0
   158
	CleanupStack::PopAndDestroy(self);	
sl@0
   159
	return NULL;
sl@0
   160
	}
sl@0
   161
sl@0
   162
CCertRetriever* CCertRetriever::NewL(X509_STORE* aStore,
sl@0
   163
						TRequestStatus& aStatus, 
sl@0
   164
						const CActiveScheduler* aActiveScheduler)
sl@0
   165
	{
sl@0
   166
	CCertRetriever* self = CCertRetriever::NewLC(aStore, aStatus, aActiveScheduler);
sl@0
   167
	if(self)
sl@0
   168
		CleanupStack::Pop();
sl@0
   169
	return self;
sl@0
   170
	}
sl@0
   171
sl@0
   172
void CCertRetriever::ConstructL()
sl@0
   173
	{
sl@0
   174
		TInt err = iFs.Connect();
sl@0
   175
		if(err != KErrNone)
sl@0
   176
			User::Leave(err);
sl@0
   177
		
sl@0
   178
 		iState = EInitializeCertStore;
sl@0
   179
 		iBuf = HBufC8::NewL(KMaxCertLength);
sl@0
   180
	}
sl@0
   181
sl@0
   182
sl@0
   183
void CCertRetriever::RunL()
sl@0
   184
	{
sl@0
   185
	// 1. All certificates retrieved.
sl@0
   186
	// 2. yes. check iActiveScheduler. if null then call User::RequestComplete(iStatus) else iActiveScheduler->Stop();
sl@0
   187
	User::LeaveIfError(iStatus.Int());
sl@0
   188
		
sl@0
   189
		switch(iState)
sl@0
   190
		{
sl@0
   191
		case EInitializeCertStore:
sl@0
   192
					OpenUnifiedCertStoreL();		
sl@0
   193
					break;
sl@0
   194
		
sl@0
   195
		case EListCerts:
sl@0
   196
					ListCertsL();	
sl@0
   197
					break;
sl@0
   198
		case EAppendCerts:
sl@0
   199
					if (!iCerts.Count()) // no certificate in store.
sl@0
   200
					{
sl@0
   201
						iState = ENoCerts;												
sl@0
   202
					}
sl@0
   203
					else
sl@0
   204
					{
sl@0
   205
						AppendCertsL();
sl@0
   206
						break;	
sl@0
   207
					}			
sl@0
   208
		       		
sl@0
   209
		case EDone:
sl@0
   210
					if (iState != ENoCerts)
sl@0
   211
					{
sl@0
   212
						ProcessCertsL();	//Process the last certificate
sl@0
   213
						iCertCount = 0;	
sl@0
   214
					}					
sl@0
   215
		case ENoCerts:
sl@0
   216
								
sl@0
   217
					if(iActiveScheduler)
sl@0
   218
					{
sl@0
   219
						if(OwnScheduler)
sl@0
   220
							iActiveScheduler->Stop();
sl@0
   221
						else
sl@0
   222
							activeSchedulerwait->AsyncStop();
sl@0
   223
						
sl@0
   224
						iFinStatus = iStatus;
sl@0
   225
					}
sl@0
   226
					else
sl@0
   227
					{
sl@0
   228
						TRequestStatus *s = &iFinStatus;
sl@0
   229
						User::RequestComplete(s, KErrNone);	
sl@0
   230
					}
sl@0
   231
					
sl@0
   232
					
sl@0
   233
					break;
sl@0
   234
		
sl@0
   235
		default:
sl@0
   236
				User::Leave(KErrNotFound);
sl@0
   237
				break;
sl@0
   238
		}
sl@0
   239
	}
sl@0
   240
sl@0
   241
sl@0
   242
void CCertRetriever::DoCancel()
sl@0
   243
	{
sl@0
   244
	}
sl@0
   245
sl@0
   246
TInt CCertRetriever::RunError(TInt aError)
sl@0
   247
	{
sl@0
   248
		//Can do some error handling here
sl@0
   249
		if(iActiveScheduler)
sl@0
   250
		{
sl@0
   251
			iActiveScheduler->Stop();
sl@0
   252
			iFinStatus = iStatus;
sl@0
   253
		}
sl@0
   254
		else
sl@0
   255
		{
sl@0
   256
			TRequestStatus *s = &iFinStatus;
sl@0
   257
			User::RequestComplete(s, aError);	
sl@0
   258
		}
sl@0
   259
		
sl@0
   260
		return KErrNone; 
sl@0
   261
	}
sl@0
   262
sl@0
   263
void CCertRetriever::RetriveCertificateL()
sl@0
   264
	{
sl@0
   265
		OpenUnifiedCertStoreL();
sl@0
   266
	}
sl@0
   267
	
sl@0
   268
	
sl@0
   269
void CCertRetriever::OpenUnifiedCertStoreL()
sl@0
   270
    {
sl@0
   271
    
sl@0
   272
    	iState = EListCerts;
sl@0
   273
    	delete iCertStore;
sl@0
   274
    	iCertStore = NULL;
sl@0
   275
    	iCertStore = CUnifiedCertStore::NewL(iFs, EFalse);
sl@0
   276
    	iCertStore->Initialize(iStatus);
sl@0
   277
    	SetActive();
sl@0
   278
    }
sl@0
   279
    
sl@0
   280
void CCertRetriever::ListCertsL()
sl@0
   281
{
sl@0
   282
    // Create filter object
sl@0
   283
   
sl@0
   284
   delete iCertFilter;
sl@0
   285
   iCertFilter = NULL;
sl@0
   286
   iCertFilter = CCertAttributeFilter::NewL();    
sl@0
   287
   iCertFilter->SetFormat(EX509Certificate);
sl@0
   288
   iCertFilter->SetOwnerType(ECACertificate);
sl@0
   289
   iCertFilter->SetUid(KTlsApplicabilityUid);
sl@0
   290
sl@0
   291
    iStatus = KRequestPending;
sl@0
   292
    SetActive();
sl@0
   293
    iCertStore->List(iCerts, *iCertFilter, iStatus);
sl@0
   294
    iState = EAppendCerts;
sl@0
   295
}
sl@0
   296
sl@0
   297
void CCertRetriever::AppendCertsL()
sl@0
   298
{
sl@0
   299
	if(iCertCount>0)	
sl@0
   300
		ProcessCertsL();
sl@0
   301
	
sl@0
   302
	CCTCertInfo *cert = iCerts[iCertCount];
sl@0
   303
		
sl@0
   304
	SetActive();			
sl@0
   305
	iStatus == KRequestPending;
sl@0
   306
		
sl@0
   307
	iCertPtr.Set( iBuf->Des() );
sl@0
   308
    iCertStore->Retrieve((*cert),iCertPtr,iStatus);
sl@0
   309
	
sl@0
   310
	iCertCount++;
sl@0
   311
	
sl@0
   312
		if(iCertCount == iCerts.Count())
sl@0
   313
			iState = EDone;	
sl@0
   314
}
sl@0
   315
sl@0
   316
sl@0
   317
void CCertRetriever::ProcessCertsL()
sl@0
   318
{
sl@0
   319
	CX509Certificate *X509Cert;
sl@0
   320
	TRAPD(error, X509Cert = CX509Certificate::NewL( iCertPtr ));
sl@0
   321
	if(error !=KErrNone)
sl@0
   322
		return;
sl@0
   323
		
sl@0
   324
	CleanupStack::PushL(X509Cert);
sl@0
   325
sl@0
   326
	X509* x509 = CX509_Initializer::CreateX509L(X509Cert);
sl@0
   327
sl@0
   328
	if(x509)
sl@0
   329
	{
sl@0
   330
		X509_STORE_add_cert(iStore,x509);
sl@0
   331
		X509_free(x509);
sl@0
   332
	}
sl@0
   333
	
sl@0
   334
	CleanupStack::PopAndDestroy(); //X509Cert
sl@0
   335
	
sl@0
   336
}