os/ossrv/genericservices/mimerecognitionfw/apmime/APMREC.CPP
changeset 0 bde4ae8d615e
     1.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     1.2 +++ b/os/ossrv/genericservices/mimerecognitionfw/apmime/APMREC.CPP	Fri Jun 15 03:10:57 2012 +0200
     1.3 @@ -0,0 +1,670 @@
     1.4 +// Copyright (c) 1997-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 +// Defines CApaDataRecognizerType the base class for concrete mime recognizers that
    1.18 +// perform the actual mime type recognitionand the CApaDataRecognizer that provides
    1.19 +// the interface to the collection of recognizers.
    1.20 +// 
    1.21 +//
    1.22 +
    1.23 +
    1.24 +#include "APMSTD.H"
    1.25 +#include "APMREC.H"
    1.26 +#include "APMPAN.H"
    1.27 +#include "APMFNDR.H"
    1.28 +
    1.29 +#ifdef USING_ECOM_RECOGS
    1.30 +#include <ecom/ecom.h>
    1.31 +#endif
    1.32 +
    1.33 +NONSHARABLE_CLASS(CDataRecognizerExtension) : public CBase
    1.34 +	{
    1.35 +public:
    1.36 +	inline static CDataRecognizerExtension* NewL() {return new(ELeave) CDataRecognizerExtension;}
    1.37 +	inline TUid DestructorUid() const {return iDtorKey;}
    1.38 +	inline TUid& DestructorUidReference() {return iDtorKey;}
    1.39 +	inline CApaDataRecognizer& DataRecognizer() {return *iDataRecognizer;}
    1.40 +	inline void SetDataRecognizer(CApaDataRecognizer& aDataRecognizer) {iDataRecognizer=&aDataRecognizer;}
    1.41 +private:
    1.42 +	inline CDataRecognizerExtension() {}
    1.43 +private:
    1.44 +	CApaDataRecognizer* iDataRecognizer; // no ownership
    1.45 +	TUid iDtorKey; // destructor key to track the instance of ECOM implementation class
    1.46 +	};
    1.47 +
    1.48 +class TDataToRecognize
    1.49 +	{
    1.50 +public:
    1.51 +	TDataToRecognize(const TDesC& aName, const TDesC8& aBuffer);
    1.52 +	TDataToRecognize(CApaDataRecognizer& aDataRecognizer, RFile& aFile, TInt aPreferredBufSize);
    1.53 +	void RecognizeL(CApaDataRecognizerType& aDataRecognizerType);
    1.54 +	void PrepareForRecognizeLoopLC();
    1.55 +	void PopOffCleanupStackAndResetAfterRecognizeLoopL();
    1.56 +private:
    1.57 +	enum
    1.58 +		{
    1.59 +		EFlagFilePositionStored	=0x00000001,
    1.60 +		EFlagLeaveIfError		=0x00000002
    1.61 +		};
    1.62 +private:
    1.63 +	static void ResetAfterRecognizeLoopL(TAny* aThis);
    1.64 +	void ResetAfterRecognizeLoopL();
    1.65 +private:
    1.66 +	TUint iFlags;
    1.67 +	const TDesC* iName; // owned if iFile!=NULL
    1.68 +	const TDesC8* iBuffer; // owned if iFile!=NULL
    1.69 +	CApaDataRecognizer* iDataRecognizer;
    1.70 +	RFile* iFile;
    1.71 +	TInt iFilePosition;
    1.72 +	TInt iPreferredBufSize;
    1.73 +	};
    1.74 +
    1.75 +//
    1.76 +// class CApaDataRecognizer
    1.77 +//
    1.78 +
    1.79 +EXPORT_C TDataRecognitionResult CApaDataRecognizer::RecognizeL(const TDesC& aName, const TDesC8& aBuffer)
    1.80 +	{
    1.81 +	TDataToRecognize dataToRecognize(aName, aBuffer);
    1.82 +	return RecognizeL(dataToRecognize);
    1.83 +	}
    1.84 +
    1.85 +EXPORT_C TDataRecognitionResult CApaDataRecognizer::RecognizeL(RFile& aFile, TInt aPreferredBufSize)
    1.86 +	{
    1.87 +	TDataToRecognize dataToRecognize(*this, aFile, aPreferredBufSize);
    1.88 +	return RecognizeL(dataToRecognize);
    1.89 +	}
    1.90 +
    1.91 +TDataRecognitionResult CApaDataRecognizer::RecognizeL(TDataToRecognize& aDataToRecognize)
    1.92 +// attempt to recognize using all recognizers
    1.93 +// keeps recognizing until either a recognizer's confidence meets or 
    1.94 +// exceeds the currently acceptable confidence, or all recognizers 
    1.95 +// have been used
    1.96 +	{
    1.97 +	iResult.Reset();
    1.98 +	aDataToRecognize.PrepareForRecognizeLoopLC();
    1.99 +	TInt storedErr=KErrNone;
   1.100 +	TInt count=iDataRecognizerList.Count();
   1.101 +	for (TInt i=0; i<count; i++)
   1.102 +		{
   1.103 +		if (DoRecognize(iDataRecognizerList[i], aDataToRecognize, TDataType(), storedErr))
   1.104 +			{
   1.105 +			break;
   1.106 +			}
   1.107 +		}
   1.108 +	aDataToRecognize.PopOffCleanupStackAndResetAfterRecognizeLoopL();
   1.109 +	// if we haven't recognized anything and there was a problem with a recognizer, propagate
   1.110 +	// the error.  n.b. if several recognizers had errors, only propagate the first error we 
   1.111 +	// encountered
   1.112 +	if (iResult.iConfidence==CApaDataRecognizerType::ENotRecognized)
   1.113 +		{
   1.114 +		User::LeaveIfError(storedErr);
   1.115 +		}
   1.116 +	return iResult;
   1.117 +	}
   1.118 +
   1.119 +EXPORT_C TBool CApaDataRecognizer::RecognizeL(const TDesC& aName, const TDesC8& aBuffer, const TDataType& aDataType)
   1.120 +	{
   1.121 +	TDataToRecognize dataToRecognize(aName, aBuffer);
   1.122 +	return RecognizeL(dataToRecognize, aDataType);
   1.123 +	}
   1.124 +
   1.125 +EXPORT_C TBool CApaDataRecognizer::RecognizeL(RFile& aFile, TInt aPreferredBufSize, const TDataType& aDataType)
   1.126 +	{
   1.127 +	TDataToRecognize dataToRecognize(*this, aFile, aPreferredBufSize);
   1.128 +	return RecognizeL(dataToRecognize, aDataType);
   1.129 +	}
   1.130 +
   1.131 +TBool CApaDataRecognizer::RecognizeL(TDataToRecognize& aDataToRecognize, const TDataType& aDataType)
   1.132 +// recognize using only recognizers which supply the aDataType
   1.133 +	{
   1.134 +	iResult.Reset();
   1.135 +	TInt count=iDataRecognizerList.Count();
   1.136 +	TInt storedErr=KErrNone;
   1.137 +	aDataToRecognize.PrepareForRecognizeLoopLC();
   1.138 +	for (TInt i=0; i<count; i++)
   1.139 +		{
   1.140 +		CApaDataRecognizerType* rec=iDataRecognizerList[i];
   1.141 +		TInt countMime=rec->MimeTypesCount();
   1.142 +		for (TInt ii=0; ii<countMime; ii++)
   1.143 +			{
   1.144 +			TDataType supportedDataType;
   1.145 +			TRAPD(err, supportedDataType=rec->SupportedDataTypeL(ii));
   1.146 +			if (err == KErrNone)
   1.147 +				{
   1.148 +				if (supportedDataType == aDataType)
   1.149 +					{
   1.150 +					if (DoRecognize(rec, aDataToRecognize, aDataType, storedErr))
   1.151 +						{
   1.152 +						aDataToRecognize.PopOffCleanupStackAndResetAfterRecognizeLoopL();
   1.153 +						return ETrue;
   1.154 +						}
   1.155 +					break;
   1.156 +					}
   1.157 +				}
   1.158 +			else if (storedErr==KErrNone)
   1.159 +				{
   1.160 +				// this is the first time we've hit an error with SupportedDataType,
   1.161 +				// so store it now and propagate later if we don't find a suitable 
   1.162 +				// recognizer
   1.163 +				storedErr=err;
   1.164 +				}
   1.165 +			}
   1.166 +		}
   1.167 +	aDataToRecognize.PopOffCleanupStackAndResetAfterRecognizeLoopL();
   1.168 +	if (iResult.iConfidence==CApaDataRecognizerType::ENotRecognized)
   1.169 +		{
   1.170 +		// a suitable recognizer wasn't found, but we may have encountered an
   1.171 +		// error with one of the recognizers we scanned n.b. only the first error
   1.172 +		// encountered is propagated
   1.173 +		User::LeaveIfError(storedErr);
   1.174 +		}
   1.175 +	return iResult.iConfidence!=CApaDataRecognizerType::ENotRecognized;
   1.176 +	}
   1.177 +
   1.178 +TBool CApaDataRecognizer::DoRecognize(CApaDataRecognizerType* aDataRecognizerType, TDataToRecognize& aDataToRecognize, const TDataType& aDataType, TInt& aError)
   1.179 +// do the recognition, if the recognizer is sufficiently confident, return ETrue
   1.180 +	{
   1.181 +	TDataRecognitionResult result;
   1.182 +	TRAPD(err, result=aDataRecognizerType->RecognizeL(aDataToRecognize));
   1.183 +	if (err != KErrNone)
   1.184 +		{
   1.185 +		// We only want to store the first error, it if is not KErrNone, it means
   1.186 +		// we have set this error previously
   1.187 +		if (aError==KErrNone)
   1.188 +			{
   1.189 +			aError=err;
   1.190 +			}
   1.191 +		return EFalse;
   1.192 +		}
   1.193 + 
   1.194 +	if (aDataType!=result.iDataType && aDataType!=TDataType())
   1.195 +		{
   1.196 +		return EFalse;
   1.197 +		}
   1.198 +	if (result.iConfidence>iResult.iConfidence)
   1.199 +		{
   1.200 +		iResult=result;
   1.201 +		return (result.iConfidence>=iAcceptedConfidence);
   1.202 +		}
   1.203 +	return EFalse;
   1.204 +	}
   1.205 +
   1.206 +EXPORT_C TInt CApaDataRecognizer::AcceptedConfidence() const
   1.207 +	{
   1.208 +	return iAcceptedConfidence;
   1.209 +	}
   1.210 +
   1.211 +EXPORT_C void CApaDataRecognizer::SetAcceptedConfidence(TInt aConfidence)
   1.212 +	{
   1.213 +	iAcceptedConfidence=aConfidence;
   1.214 +	}
   1.215 +
   1.216 +EXPORT_C CApaDataRecognizer::~CApaDataRecognizer()
   1.217 +	{
   1.218 +	iDataRecognizerList.ResetAndDestroy();
   1.219 +	iDataArray.Reset();
   1.220 +	}
   1.221 +
   1.222 +EXPORT_C CApaDataRecognizer::CApaDataRecognizer(RFs& aFs)
   1.223 +	:iFs(aFs),
   1.224 +	iMaxBufferSize(-1),
   1.225 +	iDataRecognizerList(KDataArrayGranularity),
   1.226 +	iDataArray(KDataArrayGranularity),
   1.227 +	iAcceptedConfidence(CApaDataRecognizerType::ECertain)
   1.228 +	{}
   1.229 +
   1.230 +EXPORT_C void CApaDataRecognizer::AddDataRecognizerTypeL(CApaDataRecognizerType* aDataRecognizerType)
   1.231 +// add the concrete recognizer into the list at the appropriate priority
   1.232 +	{
   1.233 +#if defined(_DEBUG)
   1.234 + 	CApaDataRecognizerType* rec=NULL;
   1.235 +	const TInt countDeb=iDataRecognizerList.Count();
   1.236 +	for (TInt ii=0; ii<countDeb; ii++)
   1.237 +		{
   1.238 +		rec=iDataRecognizerList[ii];
   1.239 + 		__ASSERT_ALWAYS(rec->TypeUid()!=aDataRecognizerType->TypeUid(), Panic(EDPanicDuplicateRecognizer));
   1.240 +		}
   1.241 +#endif // _DEBUG
   1.242 +
   1.243 +	TInt priority=aDataRecognizerType->Priority();
   1.244 +	TInt count=iDataRecognizerList.Count();
   1.245 +	for (TInt i=0; i<count; i++)
   1.246 +		{
   1.247 +		if (iDataRecognizerList[i]->Priority()<priority)
   1.248 +			{
   1.249 +			iDataRecognizerList.InsertL(aDataRecognizerType, i);
   1.250 +			aDataRecognizerType->DataRecognizerExtension()->SetDataRecognizer(*this);
   1.251 +			return;
   1.252 +			}
   1.253 +		}
   1.254 +	iDataRecognizerList.AppendL(aDataRecognizerType);
   1.255 +	aDataRecognizerType->DataRecognizerExtension()->SetDataRecognizer(*this);
   1.256 +	}
   1.257 +
   1.258 +EXPORT_C void CApaDataRecognizer::CApaDataRecognizer_Reserved_1()
   1.259 +	{}
   1.260 +
   1.261 +EXPORT_C TInt CApaDataRecognizer::RemoveDataRecognizerType(const CApaDataRecognizerType* aDataRecognizerType)
   1.262 +// remove the concrete recognizer from the list
   1.263 +// if it is not locked, if it fails return an error code
   1.264 +	{
   1.265 +	// find the recognizer in the list
   1.266 +	TInt count=iDataRecognizerList.Count();
   1.267 +	TUid uid=aDataRecognizerType->TypeUid();
   1.268 +	TInt i;
   1.269 +	for (i=0; i<count; i++)
   1.270 +		{
   1.271 +		if (iDataRecognizerList[i]->TypeUid()==uid)
   1.272 +			break;
   1.273 +		}
   1.274 +	// did we find a match
   1.275 +	if (i==count)
   1.276 +		return KErrNotFound;
   1.277 +	// is the matching recognizer locked
   1.278 +	CApaDataRecognizerType* rec=iDataRecognizerList[i];
   1.279 +	if (rec->Locked())
   1.280 +		return KErrLocked;
   1.281 +	// remove the recognizer from the list, then delete it
   1.282 +	delete rec;
   1.283 +	iDataRecognizerList.Remove(i);
   1.284 +	iDataRecognizerList.Compress();
   1.285 +	return KErrNone;
   1.286 +	}
   1.287 +
   1.288 +EXPORT_C void CApaDataRecognizer::DataTypeL(CDataTypeArray& aArray)
   1.289 +	{
   1.290 +	TInt count=iDataArray.Count();
   1.291 +	for (TInt i=0; i<count; i++)
   1.292 +		aArray.AppendL(iDataArray[i]);
   1.293 +	}
   1.294 +
   1.295 +EXPORT_C void CApaDataRecognizer::UpdateDataTypesL()
   1.296 +// rebuild the list of unique mime types
   1.297 +	{
   1.298 +	iDataArray.Reset();
   1.299 +	TInt count=iDataRecognizerList.Count();
   1.300 +	for (TInt i=0; i<count; i++)
   1.301 +		{
   1.302 +		CApaDataRecognizerType* rec=iDataRecognizerList[i];
   1.303 +		TRAP_IGNORE(rec->UpdateDataTypesL());
   1.304 +		TInt mimeCount=rec->MimeTypesCount();
   1.305 +		for (TInt ii=0; ii<mimeCount; ii++)
   1.306 +			{
   1.307 +			TRAP_IGNORE(AddDataTypeL(rec->SupportedDataTypeL(ii)));
   1.308 +			}
   1.309 +		}
   1.310 +	}
   1.311 +
   1.312 +EXPORT_C TInt CApaDataRecognizer::PreferredBufSize() const
   1.313 +// return the maximum preferred buf size
   1.314 +	{
   1.315 +	if (iMaxBufferSize < 0)
   1.316 +		{
   1.317 +		// The recognizers have been (re)loaded, so calulate the maximum buffer size.
   1.318 +		TInt count=iDataRecognizerList.Count();
   1.319 +		for (TInt i=0; i<count; i++)
   1.320 +			iMaxBufferSize=Max((TInt)iDataRecognizerList[i]->PreferredBufSize(), iMaxBufferSize);
   1.321 +		}
   1.322 +	return iMaxBufferSize;
   1.323 +	}
   1.324 +
   1.325 +EXPORT_C void CApaDataRecognizer::DestroyRecognizerList()
   1.326 +	{
   1.327 +	iDataRecognizerList.ResetAndDestroy();
   1.328 +	iDataArray.Reset();
   1.329 +	}
   1.330 +
   1.331 +void CApaDataRecognizer::AddDataTypeL(const TDataType& aDataType)
   1.332 +// add a mime type to the array if it's unique
   1.333 +	{
   1.334 +	TInt i=iDataArray.Count()-1;
   1.335 +	for (; i>=0; i--)
   1.336 +		{
   1.337 +		if (aDataType==iDataArray[i])
   1.338 +			return;
   1.339 +		}
   1.340 +	iDataArray.AppendL(aDataType);
   1.341 +	}
   1.342 +
   1.343 +//
   1.344 +// class CApaDataRecognizerType
   1.345 +//
   1.346 +
   1.347 +EXPORT_C CApaDataRecognizerType::CApaDataRecognizerType(TUid aUid, TInt aPriority)
   1.348 +	:iTypeUid(aUid),
   1.349 +	iPriority(aPriority),iDataRecognizerExtn(NULL)
   1.350 +/** Constructs the recognizer with a UID and a priority.
   1.351 +
   1.352 +Typically, a derived class constructor calls this constructor through a constructor 
   1.353 +initialization list.
   1.354 +
   1.355 +The UID is the way that a recognizer is detected by the framework.
   1.356 +
   1.357 +@param aUid A UID that identifies the recognizer.
   1.358 +@param aPriority A value that estimates the probability that the recognizer 
   1.359 +will successfully identify data. This is one of the CApaDataRecognizerType::TRecognizerPriority 
   1.360 +enumerators.
   1.361 +@see CApaDataRecognizerType::TRecognizerPriority */
   1.362 +	{}
   1.363 +
   1.364 +EXPORT_C CApaDataRecognizerType::~CApaDataRecognizerType()
   1.365 +// Destructor.
   1.366 +	{
   1.367 +#ifdef USING_ECOM_RECOGS
   1.368 +	//if ecom plugin is used destroy its implementation
   1.369 +	if(iDataRecognizerExtn!=NULL)
   1.370 +		{
   1.371 +		REComSession::DestroyedImplementation(iDataRecognizerExtn->DestructorUid());
   1.372 +		delete iDataRecognizerExtn;
   1.373 +		}
   1.374 +#endif
   1.375 +	}
   1.376 +
   1.377 +EXPORT_C void CApaDataRecognizerType::Lock()
   1.378 +/** Adds a lock to the recognizer.
   1.379 +
   1.380 +This may be called any number of times, but each call to this function must 
   1.381 +be matched by a corresponding call to Unlock() to completely unlock the recognizer.
   1.382 +
   1.383 +This function is used to prevent the recognizer DLL from being unloaded.
   1.384 +
   1.385 +@see Unlock()
   1.386 +@see Locked() */
   1.387 +	{
   1.388 +	iLock++;
   1.389 +	}
   1.390 +
   1.391 +EXPORT_C void CApaDataRecognizerType::Unlock()
   1.392 +/** Removes a lock from the recognizer.
   1.393 +
   1.394 +All calls to Lock() should be matched by a corresponding call to this function. 
   1.395 +The recognizer is not unlocked until all calls to Lock() have been matched 
   1.396 +by corresponding calls to this function.
   1.397 +
   1.398 +@see Lock()
   1.399 +@see Locked() */
   1.400 +	{
   1.401 +	if (iLock>0)
   1.402 +		iLock--;
   1.403 +	}
   1.404 +
   1.405 +EXPORT_C TUint CApaDataRecognizerType::PreferredBufSize()
   1.406 +/** Gets the size of buffer preferred for the purpose of recognizing the data type.
   1.407 +
   1.408 +Regardless of the preferred buffer size returned by an implementation of this 
   1.409 +function, the actual size used is never greater than a maximum value as set 
   1.410 +by the client of the Application Architecture server through a call to RApaLsSession::SetMaxDataBufSize().
   1.411 +
   1.412 +@return The preferred data size. The default implementation returns zero.
   1.413 +@see RApaLsSession::SetMaxDataBufSize() */
   1.414 +	{
   1.415 +	return 0;	// default to no buffer
   1.416 +				// name recognition only
   1.417 +	}
   1.418 +
   1.419 +EXPORT_C void CApaDataRecognizerType::UpdateDataTypesL()
   1.420 +/** Refreshes the list of data (MIME) types supported by this recognizer. */
   1.421 +	{
   1.422 +	}
   1.423 +
   1.424 +EXPORT_C void CApaDataRecognizerType::DoRecognizeL(const TDesC& aName, const TDesC8& aBuffer)
   1.425 +/** Implements the attempt to recognize data.
   1.426 +
   1.427 +Recognizers should provide an implementation of this function in a derived 
   1.428 +class. Note that, when the implementation recognizes data, it must put the 
   1.429 +result of the operation in the iDataType and iConfidence data members.
   1.430 +
   1.431 +Note that if more data is required than is provided in aBuffer, then 
   1.432 +CApaDataRecognizerType::FilePassedByHandleL should be called and the data 
   1.433 +read from the returned RFile (if not NULL). If this returns NULL, it may be 
   1.434 +possible to retrieve the data by calling RFile::Open() on aName, but only if 
   1.435 +aName is a legal file-name. It may be something else, such as a URL.
   1.436 +
   1.437 +The default implementation does not recognize data.
   1.438 +
   1.439 +@param aName The name of the data; typically this is a file name containing 
   1.440 +the data to be recognized. It is not necessarily a legal file-name though. It 
   1.441 +may, for example, be a URL/URI.
   1.442 +@param aBuffer A buffer containing data to be recognized; typically, this is 
   1.443 +read from the start of the file containing the data.
   1.444 +@see RecognizeL()
   1.445 +@see FilePassedByHandleL()
   1.446 +*/
   1.447 +	{
   1.448 +	(void)aName;
   1.449 +	(void)aBuffer;
   1.450 +	iDataType=TDataType();
   1.451 +	}
   1.452 +
   1.453 +EXPORT_C RFile* CApaDataRecognizerType::FilePassedByHandleL()
   1.454 +/** Returns the RFile (if any) of file to be recognized.
   1.455 +
   1.456 +This function returns the file passed by handle from the client-side (i.e. from calls to the RFile-parameter overloads of RApaLsSession's RecognizeData, RecognizeSpecificData, AppForDocument and StartDocument). The function returns NULL if the file to be recognized was not passed by handle.
   1.457 +
   1.458 +It may only be called from implementations of DoRecognizeL - indeed the purpose of this function is logically to provide an extra parameter to the virtual DoRecognizeL function. All references/pointers to the 
   1.459 +RFile object returned must be discarded when the implementation of DoRecognizeL returns.
   1.460 +
   1.461 +The RFile returned (if any) may be used by implementations of DoRecognizeL to retrieve more data than is provided in DoRecognizeL's aBuffer parameter.
   1.462 +
   1.463 +The current-position of the returned RFile is the start of the file.
   1.464 +
   1.465 +@return The file, passed by handle, to be recognized. Returns NULL if the data to be recognized was passed from the client-side by name/buffer rather than by file-handle. Ownership of the returned object is NOT transferred to the caller.
   1.466 +@see DoRecognizeL()
   1.467 +*/
   1.468 +	{
   1.469 +	RFile* const filePassedByHandle=iDataRecognizerExtn->DataRecognizer().FilePassedByHandle();
   1.470 +	if (filePassedByHandle!=NULL && (filePassedByHandle->SubSessionHandle()!=KNullHandle))
   1.471 +		{
   1.472 +		// set the current file-position to the start of the file
   1.473 +		TInt filePosition=0;
   1.474 +		User::LeaveIfError(filePassedByHandle->Seek(ESeekStart, filePosition));
   1.475 +		}
   1.476 +	return filePassedByHandle;
   1.477 +	}
   1.478 +
   1.479 +EXPORT_C TDataRecognitionResult CApaDataRecognizerType::RecognizeL(const TDesC& aName, const TDesC8& aBuffer)
   1.480 +/** Attempts to recognize data.
   1.481 +
   1.482 +This function is called by the Application Architecture server as a result 
   1.483 +of client calls to the server through an instance of RApaLsSession.
   1.484 +
   1.485 +The function calls DoRecognizeL() which implements recognition behaviour.
   1.486 +
   1.487 +@param aName The name of the data; typically this is a file name containing 
   1.488 +the data to be recognized.
   1.489 +@param aBuffer A buffer containing data to be recognized; typically, this is 
   1.490 +read from the start of the file containing the data. Implement PreferredBufSize() 
   1.491 +to define the ideal size for this buffer. Note that failure to implement PreferredBufSize() 
   1.492 +results in a default buffer size of zero.
   1.493 +@return The result of the attempt to recognize the data.
   1.494 +@see PreferredBufSize()
   1.495 +@see RApaLsSession
   1.496 +@see RApaLsSession::SetMaxDataBufSize() */
   1.497 +	{
   1.498 +	TDataToRecognize dataToRecognize(aName, aBuffer);
   1.499 +	return RecognizeL(dataToRecognize);
   1.500 +	}
   1.501 +
   1.502 +TDataRecognitionResult CApaDataRecognizerType::RecognizeL(TDataToRecognize& aDataToRecognize)
   1.503 +/**
   1.504 +@internalComponent
   1.505 +*/
   1.506 +	{
   1.507 +	iDataType=TDataType();
   1.508 +	iConfidence=ENotRecognized;
   1.509 +	aDataToRecognize.RecognizeL(*this);
   1.510 +	TDataRecognitionResult rec;
   1.511 +	rec.iDataType=iDataType;
   1.512 +	rec.iConfidence=iConfidence;
   1.513 +	return rec;
   1.514 +	}
   1.515 +
   1.516 +EXPORT_C TDataType CApaDataRecognizerType::MimeType()
   1.517 +/** Gets the data (MIME) type of the most recently recognized data.
   1.518 +
   1.519 +@return The data (MIME) type.
   1.520 +@see iDataType */
   1.521 +	{
   1.522 +	return iDataType;
   1.523 +	}
   1.524 +
   1.525 +EXPORT_C void CApaDataRecognizerType::Reserved_1()
   1.526 +// a spare empty virtual function
   1.527 +	{}
   1.528 +
   1.529 +// instantiate the ecom implementation class 
   1.530 +EXPORT_C CApaDataRecognizerType* CApaDataRecognizerType::CreateDataRecognizerL(TUid aImplUid)
   1.531 +	{
   1.532 +#if !defined(USING_ECOM_RECOGS)
   1.533 +	(void)aImplUid;
   1.534 +	return NULL;
   1.535 +#endif
   1.536 +	CDataRecognizerExtension* const dataRecognizerExtn=CDataRecognizerExtension::NewL();
   1.537 +	CleanupStack::PushL(dataRecognizerExtn);
   1.538 +
   1.539 +	CApaDataRecognizerType* const dataRecType=static_cast <CApaDataRecognizerType*>(REComSession::CreateImplementationL(aImplUid, dataRecognizerExtn->DestructorUidReference()));
   1.540 +
   1.541 +	dataRecType->iDataRecognizerExtn = dataRecognizerExtn;
   1.542 +	CleanupStack::Pop(dataRecognizerExtn);
   1.543 +
   1.544 +	return dataRecType;
   1.545 +	}
   1.546 +
   1.547 +CDataRecognizerExtension* CApaDataRecognizerType::DataRecognizerExtension()
   1.548 +/** @internalComponent */
   1.549 +	{
   1.550 +	return iDataRecognizerExtn;
   1.551 +	}
   1.552 +
   1.553 +//
   1.554 +// class TDataRecognitionResult
   1.555 +//
   1.556 +
   1.557 +EXPORT_C void TDataRecognitionResult::Reset()
   1.558 +/** Resets the data type to the default data type and sets the confidence rating 
   1.559 +to CApaDataRecognizerType::ENotRecognized.
   1.560 +
   1.561 +@see CApaDataRecognizerType::TRecognitionConfidence
   1.562 +@see TDataType */
   1.563 +	{
   1.564 +	iDataType=TDataType();
   1.565 +	iConfidence=CApaDataRecognizerType::ENotRecognized;
   1.566 +	}
   1.567 +
   1.568 +//
   1.569 +// class TDataToRecognize
   1.570 +//
   1.571 +
   1.572 +TDataToRecognize::TDataToRecognize(const TDesC& aName, const TDesC8& aBuffer)
   1.573 +	:iFlags(0),
   1.574 +	 iName(&aName),
   1.575 +	 iBuffer(&aBuffer),
   1.576 +	 iDataRecognizer(NULL),
   1.577 +	 iFile(NULL),
   1.578 +	 iFilePosition(0),
   1.579 +	 iPreferredBufSize(0)
   1.580 +	{
   1.581 +	__ASSERT_DEBUG(aName.Length()>0 || aBuffer.Length()>0, Panic(EDPanicInvalidData));
   1.582 +	}
   1.583 +
   1.584 +TDataToRecognize::TDataToRecognize(CApaDataRecognizer& aDataRecognizer, RFile& aFile, TInt aPreferredBufSize)
   1.585 +	:iFlags(0),
   1.586 +	 iName(NULL),
   1.587 +	 iBuffer(NULL),
   1.588 +	 iDataRecognizer(&aDataRecognizer),
   1.589 +	 iFile(&aFile),
   1.590 +	 iFilePosition(0),
   1.591 +	 iPreferredBufSize(aPreferredBufSize)
   1.592 +	{
   1.593 +	}
   1.594 +
   1.595 +void TDataToRecognize::RecognizeL(CApaDataRecognizerType& aDataRecognizerType)
   1.596 +	{
   1.597 +	__ASSERT_DEBUG(iName!=NULL, Panic(EDPanicNullPointer1));
   1.598 +	__ASSERT_DEBUG(iBuffer!=NULL, Panic(EDPanicNullPointer2));
   1.599 +	aDataRecognizerType.DoRecognizeL(*iName, *iBuffer);
   1.600 +	}
   1.601 +
   1.602 +void TDataToRecognize::PrepareForRecognizeLoopLC()
   1.603 +	{
   1.604 +	CleanupStack::PushL(TCleanupItem(ResetAfterRecognizeLoopL, this));
   1.605 +	if (iFile!=NULL)
   1.606 +		{
   1.607 +		// set the file-pointer returned by CApaDataRecognizerType::FilePassedByHandleL
   1.608 +		iDataRecognizer->SetFilePassedByHandle(iFile);
   1.609 +
   1.610 +		// store current-position in iFilePosition
   1.611 +		__ASSERT_DEBUG((iFlags&EFlagFilePositionStored)==0, Panic(EDPanicBadFlagState1));
   1.612 +		iFlags|=EFlagFilePositionStored;
   1.613 +		iFilePosition=0;
   1.614 +		User::LeaveIfError(iFile->Seek(ESeekCurrent, iFilePosition));
   1.615 +
   1.616 +		// read a buffer of the appropriate length from the start of the file
   1.617 +		__ASSERT_DEBUG(iBuffer==NULL, Panic(EDPanicNullPointerExpected1));
   1.618 +		TInt filePosition=0;
   1.619 +		User::LeaveIfError(iFile->Seek(ESeekStart, filePosition));
   1.620 +		HBufC8* const buffer=HBufC8::NewL(iPreferredBufSize);
   1.621 +		iBuffer=buffer; // iBuffer now "owns" this (as long as iFile!=NULL, which is true)
   1.622 +		TPtr8 buffer_asWritable(buffer->Des());
   1.623 +		User::LeaveIfError(iFile->Read(buffer_asWritable));
   1.624 +
   1.625 +		// prepend "::" to the file-name to be passed to DoRecognizeL to make it an illegal file-name - this
   1.626 +		// is to prevent any behavioural break since implementers of DoRecognizeL may assume that if aName is
   1.627 +		// *not* an illegal file-name they can pass it to RFile::Open, which will not be possible if we're
   1.628 +		// recognizing by file-handle, both because of data-caging (the file may not live in a directory that
   1.629 +		// the Apparc process has permission to open it from) and because aName does not contain the full path
   1.630 +		__ASSERT_DEBUG(iName==NULL, Panic(EDPanicNullPointerExpected2));
   1.631 +		TDes* const name=new(ELeave) TFileName;
   1.632 +		iName=name; // iName now "owns" this (as long as iFile!=NULL, which is true)
   1.633 +		User::LeaveIfError(iFile->Name(*name));
   1.634 +		_LIT(KLitIllegalifier, "::");
   1.635 +		name->Insert(0, KLitIllegalifier);
   1.636 +		}
   1.637 +	}
   1.638 +
   1.639 +void TDataToRecognize::PopOffCleanupStackAndResetAfterRecognizeLoopL()
   1.640 +	{
   1.641 +	iFlags|=EFlagLeaveIfError; // we don't want to leave from ResetAfterRecognizeLoopL if we're being cleaned up from the clean-up stack because of a leave, but if we're leaving because of a CleanupStack::PopAndDestroy then it's okay to leave
   1.642 +	CleanupStack::PopAndDestroy(this); // calls ResetAfterRecognizeLoopL
   1.643 +	}
   1.644 +
   1.645 +void TDataToRecognize::ResetAfterRecognizeLoopL(TAny* aThis)
   1.646 +	{ // static
   1.647 +	STATIC_CAST(TDataToRecognize*, aThis)->ResetAfterRecognizeLoopL();
   1.648 +	}
   1.649 +
   1.650 +void TDataToRecognize::ResetAfterRecognizeLoopL()
   1.651 +	{
   1.652 +	TInt error=KErrNone;
   1.653 +	if (iFile!=NULL)
   1.654 +		{
   1.655 +		// NULLify the file-pointer returned by CApaDataRecognizerType::FilePassedByHandleL
   1.656 +		iDataRecognizer->SetFilePassedByHandle(NULL);
   1.657 +
   1.658 +		// reset the current-position of the file
   1.659 +		__ASSERT_DEBUG(iFlags&EFlagFilePositionStored, Panic(EDPanicBadFlagState2));
   1.660 +		error=iFile->Seek(ESeekStart, iFilePosition);
   1.661 +		iFlags&=~EFlagFilePositionStored;
   1.662 +
   1.663 +		// delete the objects that are owned if iFile!=NULL (which is true)
   1.664 +		delete iName;
   1.665 +		delete iBuffer;
   1.666 +		}
   1.667 +	if (iFlags&EFlagLeaveIfError)
   1.668 +		{
   1.669 +		iFlags&=~EFlagLeaveIfError; // probably not necessary, just defensive programming
   1.670 +		User::LeaveIfError(error);
   1.671 +		}
   1.672 +	}
   1.673 +