Update contrib.
1 // Copyright (c) 1997-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
14 // Defines CApaDataRecognizerType the base class for concrete mime recognizers that
15 // perform the actual mime type recognitionand the CApaDataRecognizer that provides
16 // the interface to the collection of recognizers.
26 #ifdef USING_ECOM_RECOGS
27 #include <ecom/ecom.h>
30 NONSHARABLE_CLASS(CDataRecognizerExtension) : public CBase
33 inline static CDataRecognizerExtension* NewL() {return new(ELeave) CDataRecognizerExtension;}
34 inline TUid DestructorUid() const {return iDtorKey;}
35 inline TUid& DestructorUidReference() {return iDtorKey;}
36 inline CApaDataRecognizer& DataRecognizer() {return *iDataRecognizer;}
37 inline void SetDataRecognizer(CApaDataRecognizer& aDataRecognizer) {iDataRecognizer=&aDataRecognizer;}
39 inline CDataRecognizerExtension() {}
41 CApaDataRecognizer* iDataRecognizer; // no ownership
42 TUid iDtorKey; // destructor key to track the instance of ECOM implementation class
45 class TDataToRecognize
48 TDataToRecognize(const TDesC& aName, const TDesC8& aBuffer);
49 TDataToRecognize(CApaDataRecognizer& aDataRecognizer, RFile& aFile, TInt aPreferredBufSize);
50 void RecognizeL(CApaDataRecognizerType& aDataRecognizerType);
51 void PrepareForRecognizeLoopLC();
52 void PopOffCleanupStackAndResetAfterRecognizeLoopL();
56 EFlagFilePositionStored =0x00000001,
57 EFlagLeaveIfError =0x00000002
60 static void ResetAfterRecognizeLoopL(TAny* aThis);
61 void ResetAfterRecognizeLoopL();
64 const TDesC* iName; // owned if iFile!=NULL
65 const TDesC8* iBuffer; // owned if iFile!=NULL
66 CApaDataRecognizer* iDataRecognizer;
69 TInt iPreferredBufSize;
73 // class CApaDataRecognizer
76 EXPORT_C TDataRecognitionResult CApaDataRecognizer::RecognizeL(const TDesC& aName, const TDesC8& aBuffer)
78 TDataToRecognize dataToRecognize(aName, aBuffer);
79 return RecognizeL(dataToRecognize);
82 EXPORT_C TDataRecognitionResult CApaDataRecognizer::RecognizeL(RFile& aFile, TInt aPreferredBufSize)
84 TDataToRecognize dataToRecognize(*this, aFile, aPreferredBufSize);
85 return RecognizeL(dataToRecognize);
88 TDataRecognitionResult CApaDataRecognizer::RecognizeL(TDataToRecognize& aDataToRecognize)
89 // attempt to recognize using all recognizers
90 // keeps recognizing until either a recognizer's confidence meets or
91 // exceeds the currently acceptable confidence, or all recognizers
95 aDataToRecognize.PrepareForRecognizeLoopLC();
96 TInt storedErr=KErrNone;
97 TInt count=iDataRecognizerList.Count();
98 for (TInt i=0; i<count; i++)
100 if (DoRecognize(iDataRecognizerList[i], aDataToRecognize, TDataType(), storedErr))
105 aDataToRecognize.PopOffCleanupStackAndResetAfterRecognizeLoopL();
106 // if we haven't recognized anything and there was a problem with a recognizer, propagate
107 // the error. n.b. if several recognizers had errors, only propagate the first error we
109 if (iResult.iConfidence==CApaDataRecognizerType::ENotRecognized)
111 User::LeaveIfError(storedErr);
116 EXPORT_C TBool CApaDataRecognizer::RecognizeL(const TDesC& aName, const TDesC8& aBuffer, const TDataType& aDataType)
118 TDataToRecognize dataToRecognize(aName, aBuffer);
119 return RecognizeL(dataToRecognize, aDataType);
122 EXPORT_C TBool CApaDataRecognizer::RecognizeL(RFile& aFile, TInt aPreferredBufSize, const TDataType& aDataType)
124 TDataToRecognize dataToRecognize(*this, aFile, aPreferredBufSize);
125 return RecognizeL(dataToRecognize, aDataType);
128 TBool CApaDataRecognizer::RecognizeL(TDataToRecognize& aDataToRecognize, const TDataType& aDataType)
129 // recognize using only recognizers which supply the aDataType
132 TInt count=iDataRecognizerList.Count();
133 TInt storedErr=KErrNone;
134 aDataToRecognize.PrepareForRecognizeLoopLC();
135 for (TInt i=0; i<count; i++)
137 CApaDataRecognizerType* rec=iDataRecognizerList[i];
138 TInt countMime=rec->MimeTypesCount();
139 for (TInt ii=0; ii<countMime; ii++)
141 TDataType supportedDataType;
142 TRAPD(err, supportedDataType=rec->SupportedDataTypeL(ii));
145 if (supportedDataType == aDataType)
147 if (DoRecognize(rec, aDataToRecognize, aDataType, storedErr))
149 aDataToRecognize.PopOffCleanupStackAndResetAfterRecognizeLoopL();
155 else if (storedErr==KErrNone)
157 // this is the first time we've hit an error with SupportedDataType,
158 // so store it now and propagate later if we don't find a suitable
164 aDataToRecognize.PopOffCleanupStackAndResetAfterRecognizeLoopL();
165 if (iResult.iConfidence==CApaDataRecognizerType::ENotRecognized)
167 // a suitable recognizer wasn't found, but we may have encountered an
168 // error with one of the recognizers we scanned n.b. only the first error
169 // encountered is propagated
170 User::LeaveIfError(storedErr);
172 return iResult.iConfidence!=CApaDataRecognizerType::ENotRecognized;
175 TBool CApaDataRecognizer::DoRecognize(CApaDataRecognizerType* aDataRecognizerType, TDataToRecognize& aDataToRecognize, const TDataType& aDataType, TInt& aError)
176 // do the recognition, if the recognizer is sufficiently confident, return ETrue
178 TDataRecognitionResult result;
179 TRAPD(err, result=aDataRecognizerType->RecognizeL(aDataToRecognize));
182 // We only want to store the first error, it if is not KErrNone, it means
183 // we have set this error previously
184 if (aError==KErrNone)
191 if (aDataType!=result.iDataType && aDataType!=TDataType())
195 if (result.iConfidence>iResult.iConfidence)
198 return (result.iConfidence>=iAcceptedConfidence);
203 EXPORT_C TInt CApaDataRecognizer::AcceptedConfidence() const
205 return iAcceptedConfidence;
208 EXPORT_C void CApaDataRecognizer::SetAcceptedConfidence(TInt aConfidence)
210 iAcceptedConfidence=aConfidence;
213 EXPORT_C CApaDataRecognizer::~CApaDataRecognizer()
215 iDataRecognizerList.ResetAndDestroy();
219 EXPORT_C CApaDataRecognizer::CApaDataRecognizer(RFs& aFs)
222 iDataRecognizerList(KDataArrayGranularity),
223 iDataArray(KDataArrayGranularity),
224 iAcceptedConfidence(CApaDataRecognizerType::ECertain)
227 EXPORT_C void CApaDataRecognizer::AddDataRecognizerTypeL(CApaDataRecognizerType* aDataRecognizerType)
228 // add the concrete recognizer into the list at the appropriate priority
231 CApaDataRecognizerType* rec=NULL;
232 const TInt countDeb=iDataRecognizerList.Count();
233 for (TInt ii=0; ii<countDeb; ii++)
235 rec=iDataRecognizerList[ii];
236 __ASSERT_ALWAYS(rec->TypeUid()!=aDataRecognizerType->TypeUid(), Panic(EDPanicDuplicateRecognizer));
240 TInt priority=aDataRecognizerType->Priority();
241 TInt count=iDataRecognizerList.Count();
242 for (TInt i=0; i<count; i++)
244 if (iDataRecognizerList[i]->Priority()<priority)
246 iDataRecognizerList.InsertL(aDataRecognizerType, i);
247 aDataRecognizerType->DataRecognizerExtension()->SetDataRecognizer(*this);
251 iDataRecognizerList.AppendL(aDataRecognizerType);
252 aDataRecognizerType->DataRecognizerExtension()->SetDataRecognizer(*this);
255 EXPORT_C void CApaDataRecognizer::CApaDataRecognizer_Reserved_1()
258 EXPORT_C TInt CApaDataRecognizer::RemoveDataRecognizerType(const CApaDataRecognizerType* aDataRecognizerType)
259 // remove the concrete recognizer from the list
260 // if it is not locked, if it fails return an error code
262 // find the recognizer in the list
263 TInt count=iDataRecognizerList.Count();
264 TUid uid=aDataRecognizerType->TypeUid();
266 for (i=0; i<count; i++)
268 if (iDataRecognizerList[i]->TypeUid()==uid)
271 // did we find a match
274 // is the matching recognizer locked
275 CApaDataRecognizerType* rec=iDataRecognizerList[i];
278 // remove the recognizer from the list, then delete it
280 iDataRecognizerList.Remove(i);
281 iDataRecognizerList.Compress();
285 EXPORT_C void CApaDataRecognizer::DataTypeL(CDataTypeArray& aArray)
287 TInt count=iDataArray.Count();
288 for (TInt i=0; i<count; i++)
289 aArray.AppendL(iDataArray[i]);
292 EXPORT_C void CApaDataRecognizer::UpdateDataTypesL()
293 // rebuild the list of unique mime types
296 TInt count=iDataRecognizerList.Count();
297 for (TInt i=0; i<count; i++)
299 CApaDataRecognizerType* rec=iDataRecognizerList[i];
300 TRAP_IGNORE(rec->UpdateDataTypesL());
301 TInt mimeCount=rec->MimeTypesCount();
302 for (TInt ii=0; ii<mimeCount; ii++)
304 TRAP_IGNORE(AddDataTypeL(rec->SupportedDataTypeL(ii)));
309 EXPORT_C TInt CApaDataRecognizer::PreferredBufSize() const
310 // return the maximum preferred buf size
312 if (iMaxBufferSize < 0)
314 // The recognizers have been (re)loaded, so calulate the maximum buffer size.
315 TInt count=iDataRecognizerList.Count();
316 for (TInt i=0; i<count; i++)
317 iMaxBufferSize=Max((TInt)iDataRecognizerList[i]->PreferredBufSize(), iMaxBufferSize);
319 return iMaxBufferSize;
322 EXPORT_C void CApaDataRecognizer::DestroyRecognizerList()
324 iDataRecognizerList.ResetAndDestroy();
328 void CApaDataRecognizer::AddDataTypeL(const TDataType& aDataType)
329 // add a mime type to the array if it's unique
331 TInt i=iDataArray.Count()-1;
334 if (aDataType==iDataArray[i])
337 iDataArray.AppendL(aDataType);
341 // class CApaDataRecognizerType
344 EXPORT_C CApaDataRecognizerType::CApaDataRecognizerType(TUid aUid, TInt aPriority)
346 iPriority(aPriority),iDataRecognizerExtn(NULL)
347 /** Constructs the recognizer with a UID and a priority.
349 Typically, a derived class constructor calls this constructor through a constructor
352 The UID is the way that a recognizer is detected by the framework.
354 @param aUid A UID that identifies the recognizer.
355 @param aPriority A value that estimates the probability that the recognizer
356 will successfully identify data. This is one of the CApaDataRecognizerType::TRecognizerPriority
358 @see CApaDataRecognizerType::TRecognizerPriority */
361 EXPORT_C CApaDataRecognizerType::~CApaDataRecognizerType()
364 #ifdef USING_ECOM_RECOGS
365 //if ecom plugin is used destroy its implementation
366 if(iDataRecognizerExtn!=NULL)
368 REComSession::DestroyedImplementation(iDataRecognizerExtn->DestructorUid());
369 delete iDataRecognizerExtn;
374 EXPORT_C void CApaDataRecognizerType::Lock()
375 /** Adds a lock to the recognizer.
377 This may be called any number of times, but each call to this function must
378 be matched by a corresponding call to Unlock() to completely unlock the recognizer.
380 This function is used to prevent the recognizer DLL from being unloaded.
388 EXPORT_C void CApaDataRecognizerType::Unlock()
389 /** Removes a lock from the recognizer.
391 All calls to Lock() should be matched by a corresponding call to this function.
392 The recognizer is not unlocked until all calls to Lock() have been matched
393 by corresponding calls to this function.
402 EXPORT_C TUint CApaDataRecognizerType::PreferredBufSize()
403 /** Gets the size of buffer preferred for the purpose of recognizing the data type.
405 Regardless of the preferred buffer size returned by an implementation of this
406 function, the actual size used is never greater than a maximum value as set
407 by the client of the Application Architecture server through a call to RApaLsSession::SetMaxDataBufSize().
409 @return The preferred data size. The default implementation returns zero.
410 @see RApaLsSession::SetMaxDataBufSize() */
412 return 0; // default to no buffer
413 // name recognition only
416 EXPORT_C void CApaDataRecognizerType::UpdateDataTypesL()
417 /** Refreshes the list of data (MIME) types supported by this recognizer. */
421 EXPORT_C void CApaDataRecognizerType::DoRecognizeL(const TDesC& aName, const TDesC8& aBuffer)
422 /** Implements the attempt to recognize data.
424 Recognizers should provide an implementation of this function in a derived
425 class. Note that, when the implementation recognizes data, it must put the
426 result of the operation in the iDataType and iConfidence data members.
428 Note that if more data is required than is provided in aBuffer, then
429 CApaDataRecognizerType::FilePassedByHandleL should be called and the data
430 read from the returned RFile (if not NULL). If this returns NULL, it may be
431 possible to retrieve the data by calling RFile::Open() on aName, but only if
432 aName is a legal file-name. It may be something else, such as a URL.
434 The default implementation does not recognize data.
436 @param aName The name of the data; typically this is a file name containing
437 the data to be recognized. It is not necessarily a legal file-name though. It
438 may, for example, be a URL/URI.
439 @param aBuffer A buffer containing data to be recognized; typically, this is
440 read from the start of the file containing the data.
442 @see FilePassedByHandleL()
447 iDataType=TDataType();
450 EXPORT_C RFile* CApaDataRecognizerType::FilePassedByHandleL()
451 /** Returns the RFile (if any) of file to be recognized.
453 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.
455 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
456 RFile object returned must be discarded when the implementation of DoRecognizeL returns.
458 The RFile returned (if any) may be used by implementations of DoRecognizeL to retrieve more data than is provided in DoRecognizeL's aBuffer parameter.
460 The current-position of the returned RFile is the start of the file.
462 @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.
466 RFile* const filePassedByHandle=iDataRecognizerExtn->DataRecognizer().FilePassedByHandle();
467 if (filePassedByHandle!=NULL && (filePassedByHandle->SubSessionHandle()!=KNullHandle))
469 // set the current file-position to the start of the file
471 User::LeaveIfError(filePassedByHandle->Seek(ESeekStart, filePosition));
473 return filePassedByHandle;
476 EXPORT_C TDataRecognitionResult CApaDataRecognizerType::RecognizeL(const TDesC& aName, const TDesC8& aBuffer)
477 /** Attempts to recognize data.
479 This function is called by the Application Architecture server as a result
480 of client calls to the server through an instance of RApaLsSession.
482 The function calls DoRecognizeL() which implements recognition behaviour.
484 @param aName The name of the data; typically this is a file name containing
485 the data to be recognized.
486 @param aBuffer A buffer containing data to be recognized; typically, this is
487 read from the start of the file containing the data. Implement PreferredBufSize()
488 to define the ideal size for this buffer. Note that failure to implement PreferredBufSize()
489 results in a default buffer size of zero.
490 @return The result of the attempt to recognize the data.
491 @see PreferredBufSize()
493 @see RApaLsSession::SetMaxDataBufSize() */
495 TDataToRecognize dataToRecognize(aName, aBuffer);
496 return RecognizeL(dataToRecognize);
499 TDataRecognitionResult CApaDataRecognizerType::RecognizeL(TDataToRecognize& aDataToRecognize)
504 iDataType=TDataType();
505 iConfidence=ENotRecognized;
506 aDataToRecognize.RecognizeL(*this);
507 TDataRecognitionResult rec;
508 rec.iDataType=iDataType;
509 rec.iConfidence=iConfidence;
513 EXPORT_C TDataType CApaDataRecognizerType::MimeType()
514 /** Gets the data (MIME) type of the most recently recognized data.
516 @return The data (MIME) type.
522 EXPORT_C void CApaDataRecognizerType::Reserved_1()
523 // a spare empty virtual function
526 // instantiate the ecom implementation class
527 EXPORT_C CApaDataRecognizerType* CApaDataRecognizerType::CreateDataRecognizerL(TUid aImplUid)
529 #if !defined(USING_ECOM_RECOGS)
533 CDataRecognizerExtension* const dataRecognizerExtn=CDataRecognizerExtension::NewL();
534 CleanupStack::PushL(dataRecognizerExtn);
536 CApaDataRecognizerType* const dataRecType=static_cast <CApaDataRecognizerType*>(REComSession::CreateImplementationL(aImplUid, dataRecognizerExtn->DestructorUidReference()));
538 dataRecType->iDataRecognizerExtn = dataRecognizerExtn;
539 CleanupStack::Pop(dataRecognizerExtn);
544 CDataRecognizerExtension* CApaDataRecognizerType::DataRecognizerExtension()
545 /** @internalComponent */
547 return iDataRecognizerExtn;
551 // class TDataRecognitionResult
554 EXPORT_C void TDataRecognitionResult::Reset()
555 /** Resets the data type to the default data type and sets the confidence rating
556 to CApaDataRecognizerType::ENotRecognized.
558 @see CApaDataRecognizerType::TRecognitionConfidence
561 iDataType=TDataType();
562 iConfidence=CApaDataRecognizerType::ENotRecognized;
566 // class TDataToRecognize
569 TDataToRecognize::TDataToRecognize(const TDesC& aName, const TDesC8& aBuffer)
573 iDataRecognizer(NULL),
578 __ASSERT_DEBUG(aName.Length()>0 || aBuffer.Length()>0, Panic(EDPanicInvalidData));
581 TDataToRecognize::TDataToRecognize(CApaDataRecognizer& aDataRecognizer, RFile& aFile, TInt aPreferredBufSize)
585 iDataRecognizer(&aDataRecognizer),
588 iPreferredBufSize(aPreferredBufSize)
592 void TDataToRecognize::RecognizeL(CApaDataRecognizerType& aDataRecognizerType)
594 __ASSERT_DEBUG(iName!=NULL, Panic(EDPanicNullPointer1));
595 __ASSERT_DEBUG(iBuffer!=NULL, Panic(EDPanicNullPointer2));
596 aDataRecognizerType.DoRecognizeL(*iName, *iBuffer);
599 void TDataToRecognize::PrepareForRecognizeLoopLC()
601 CleanupStack::PushL(TCleanupItem(ResetAfterRecognizeLoopL, this));
604 // set the file-pointer returned by CApaDataRecognizerType::FilePassedByHandleL
605 iDataRecognizer->SetFilePassedByHandle(iFile);
607 // store current-position in iFilePosition
608 __ASSERT_DEBUG((iFlags&EFlagFilePositionStored)==0, Panic(EDPanicBadFlagState1));
609 iFlags|=EFlagFilePositionStored;
611 User::LeaveIfError(iFile->Seek(ESeekCurrent, iFilePosition));
613 // read a buffer of the appropriate length from the start of the file
614 __ASSERT_DEBUG(iBuffer==NULL, Panic(EDPanicNullPointerExpected1));
616 User::LeaveIfError(iFile->Seek(ESeekStart, filePosition));
617 HBufC8* const buffer=HBufC8::NewL(iPreferredBufSize);
618 iBuffer=buffer; // iBuffer now "owns" this (as long as iFile!=NULL, which is true)
619 TPtr8 buffer_asWritable(buffer->Des());
620 User::LeaveIfError(iFile->Read(buffer_asWritable));
622 // prepend "::" to the file-name to be passed to DoRecognizeL to make it an illegal file-name - this
623 // is to prevent any behavioural break since implementers of DoRecognizeL may assume that if aName is
624 // *not* an illegal file-name they can pass it to RFile::Open, which will not be possible if we're
625 // recognizing by file-handle, both because of data-caging (the file may not live in a directory that
626 // the Apparc process has permission to open it from) and because aName does not contain the full path
627 __ASSERT_DEBUG(iName==NULL, Panic(EDPanicNullPointerExpected2));
628 TDes* const name=new(ELeave) TFileName;
629 iName=name; // iName now "owns" this (as long as iFile!=NULL, which is true)
630 User::LeaveIfError(iFile->Name(*name));
631 _LIT(KLitIllegalifier, "::");
632 name->Insert(0, KLitIllegalifier);
636 void TDataToRecognize::PopOffCleanupStackAndResetAfterRecognizeLoopL()
638 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
639 CleanupStack::PopAndDestroy(this); // calls ResetAfterRecognizeLoopL
642 void TDataToRecognize::ResetAfterRecognizeLoopL(TAny* aThis)
644 STATIC_CAST(TDataToRecognize*, aThis)->ResetAfterRecognizeLoopL();
647 void TDataToRecognize::ResetAfterRecognizeLoopL()
652 // NULLify the file-pointer returned by CApaDataRecognizerType::FilePassedByHandleL
653 iDataRecognizer->SetFilePassedByHandle(NULL);
655 // reset the current-position of the file
656 __ASSERT_DEBUG(iFlags&EFlagFilePositionStored, Panic(EDPanicBadFlagState2));
657 error=iFile->Seek(ESeekStart, iFilePosition);
658 iFlags&=~EFlagFilePositionStored;
660 // delete the objects that are owned if iFile!=NULL (which is true)
664 if (iFlags&EFlagLeaveIfError)
666 iFlags&=~EFlagLeaveIfError; // probably not necessary, just defensive programming
667 User::LeaveIfError(error);