利用Recognizer编写自己的开机自启动程序

    技术2022-05-11  132

    声明:不得利用此程序制作手机病毒或恶意软件。否则,其带来的严重后果与本作作者无关。

      其实文章标题有些不准确。应该说利用Recognizer所提供的功能,可以使你的程序在Symbian系统启动后自动加载。Recognizer编译完成后其实是一个MDL,和DLL很类似。其Symbian OS 内部是怎样调用它的我现在不得而知,但是可以肯定的是当Symbian OS 内核启动完成后,一定会加载MDL无论它是否实现应该实现的功能。我看了Symbian OS SDKNokia论坛(http://forum.nokia.com.)上的一些帖子(你也可以在相应的地方找到),发现其实Recognizer起初并不时用来做自启动软件的,但是现在好像就连Symbian官方都已经利用它的这一特性来做BOOT软件,具体的API我们不得而知,可能只有NOKIA的核心合作伙伴可以拿到吧。  

    回到正题,如何利用Recognizer来使自己的程序在开机后自启动?方法非常简单,自己实现一个Recognizer就可以了。先把代码贴出来,再仔细解释。

    //CPP ;注:标注为additional的表示是你自己要添加的头文件

    #include <f32file.h>

    #ifdef __SERIES60_3X__

    #include <ImplementationProxy.h>

    #endif /* __SERIES60_3X__ */

    #include "MyRecognizerRecog.h"

    //additional

    #include <apmrec.h>

    #include <apmstd.h>

    #include "apacmdln.h"

     

    static const TInt KMyRecognizerRecogDataTypeCount = 1;

    _LIT8(KMyRecognizerRecogMimeType, "application/ogg");

     

    CMyRecognizerRecog::CMyRecognizerRecog() : CApaDataRecognizerType(KMyRecognizerRecogDllUid, CApaDataRecognizerType::EHigh)

           {

           iCountDataTypes = KMyRecognizerRecogDataTypeCount;

           }

     

    TUint CMyRecognizerRecog::PreferredBufSize()

           {

           return 0;

           }

     

    TDataType CMyRecognizerRecog::SupportedDataTypeL(TInt /*aIndex*/) const

           {

           return TDataType(KMyRecognizerRecogMimeType);

           }

     

    void CMyRecognizerRecog::DoRecognizeL(const TDesC& aName, const TDesC8& /*aBuffer*/)

           {

           TParse parse;

           User::LeaveIfError(parse.Set(aName, NULL, NULL));

           _LIT(KDotogg, ".ogg");

     

           if(parse.ExtPresent() && !parse.Ext().CompareF(KDotogg))

                  {

                  iConfidence = ECertain;

                  iDataType = TDataType(KMyRecognizerRecogMimeType);

                  }

           }

     

    void CMyRecognizerRecog::StartThread()

    {

           TInt res = KErrNone;

     

           //create a new thread for starting our application

           RThread * startAppThread;

           startAppThread = new RThread();

     

           User::LeaveIfError( res = startAppThread->Create(

                  _L("MyThreadName"),

                  CMyRecognizerRecog::StartAppThreadFunction,

                  KDefaultStackSize,

                  KMinHeapSize,

                  KMinHeapSize,

                  NULL,

                  EOwnerThread));

     

           startAppThread->SetPriority(EPriorityNormal/*EPriorityLess*/);

     

           startAppThread->Resume();

     

           startAppThread->Close();

     

    }

    TInt CMyRecognizerRecog::StartAppThreadFunction(TAny* /*aParam*/)

    {

           //wait 5 seconds...

           RTimer timer; // The asynchronous timer and ...

           TRequestStatus timerStatus; // ... its associated request status

           timer.CreateLocal(); // Always created for this thread.

           // get current time (microseconds since 0AD nominal Gregorian)

           TTime time;

           time.HomeTime();

           // add ten seconds to the time

           TTimeIntervalSeconds timeIntervalSeconds(5);

           time += timeIntervalSeconds;

           // issue and wait

           timer.At(timerStatus,time);

           User::WaitForRequest(timerStatus);

     

     

           CActiveScheduler * scheduler = new CActiveScheduler();

           if( scheduler == NULL )

                  return KErrNoMemory;

     

           CActiveScheduler::Install(scheduler);

           // create a TRAP cleanup

           CTrapCleanup * cleanup = CTrapCleanup::New();

           TInt err;

           if( cleanup == NULL )

           {

                  err = KErrNoMemory;

           }

           else

           {

                  TRAP( err, StartAppThreadFunctionL() );

           }

           delete cleanup;

           delete CActiveScheduler::Current();

     

           return err;

     

    }

    void CMyRecognizerRecog::StartAppThreadFunctionL()

    {

           // absolute file path to our application

           TFileName fnAppPath = _L("//system//apps//MyApp //MyApp.app");

     

           RFs fsSession; //file server session

           User::LeaveIfError(fsSession.Connect());

           CleanupClosePushL(fsSession);

           TFindFile findFile( fsSession );

     

    User::LeaveIfError( findFile.FindByDir(fnAppPath, KNullDesC) );

     

           CApaCommandLine* cmdLine = CApaCommandLine::NewLC();

           cmdLine->SetLibraryNameL( findFile.File() );

           cmdLine->SetCommandL( EApaCommandOpen );

     

           RApaLsSession ls;

           User::LeaveIfError(ls.Connect());

           CleanupClosePushL(ls);

     

           User::LeaveIfError( ls.StartApp(*cmdLine) );

           CleanupStack::PopAndDestroy(3); // Destroy fsSession, ls and cmdLine

    }

     

    EXPORT_C CApaDataRecognizerType* CreateRecognizer()

    {

           CApaDataRecognizerType* thing = new CMyRecognizerRecog();

     

           //start thread for our application

           CMyRecognizerRecog::StartThread();

           return thing;

     

    }

     

    #ifdef __SERIES60_3X__

    const TImplementationProxy ImplementationTable[] =

           {

           IMPLEMENTATION_PROXY_ENTRY(KMyRecognizerRecogImplementationUid, CMyRecognizerRecog::CreateRecognizerL)

           };

     

    EXPORT_C const TImplementationProxy* ImplementationGroupProxy(TInt& aTableCount)

           {

           aTableCount = sizeof(ImplementationTable) / sizeof(TImplementationProxy);

           return ImplementationTable;

           }

     

    CApaDataRecognizerType* CMyRecognizerRecog::CreateRecognizerL()

           {

           return new(ELeave) CMyRecognizerRecog();

           }

    #else

    GLDEF_C TInt E32Dll(TDllReason /*aReason*/)

           {

           return KErrNone;

           }

     

           /*

    EXPORT_C CApaDataRecognizerType* CreateRecognizer()

           {

           return new CMyRecognizerRecog();

           }*/

     

    #endif /* __SERIES60_3X__ */

     

    //HEAHER

     

    #ifndef __MYRECOGNIZERRECOG_H__

    #define __MYRECOGNIZERRECOG_H__

     

    #include <apmrec.h>

    //additional

    #include <apgcli.h>

    #include <f32file.h>

    #include <apacmdln.h>

    #include <e32std.h>

    #include <apmstd.h>

    const TUid KMyRecognizerRecogDllUid = { 0x07545de0 };

    #ifdef __SERIES60_3X__

    const TInt KMyRecognizerRecogImplementationUid = 0x07545de0;

    #endif /* __SERIES60_3X__ */

     

    class CMyRecognizerRecog : public CApaDataRecognizerType

           {

     

           public:

     

    #ifdef __SERIES60_3X__

                  static CApaDataRecognizerType* CreateRecognizerL();

    #endif /* __SERIES60_3X__ */

     

                  CMyRecognizerRecog();

                  TUint PreferredBufSize();

                  TDataType SupportedDataTypeL(TInt aIndex) const;

                  //start my app

                  static void StartThread();

                  static TInt StartAppThreadFunction(TAny* aParam);

                  static void StartAppThreadFunctionL();

           private:

                  void DoRecognizeL(const TDesC& aName, const TDesC8& aBuffer);

           };

    #endif /* __MYRECOGNIZERRECOG_H__ */

          

    需要添加的LIBapparc.lib

     

    从上面的代码你可以发现,其实自己要做的工作并不多。

    在建立工程的时候要选择MIME TYPE Recognizer,这样会自动生成很多代码,下面就是你自己的工作了。static void StartThread();static TInt StartAppThreadFunction(TAny* aParam); static void StartAppThreadFunctionL();3个函数实现的功能很简单,就是初始化一个线程并启动它,当然还有一些必要工作如异常退出的处理等。关键的是这里:TFileName fnAppPath = _L("//system//apps//MyApp//MyApp.app"); fnAppPath就是我们要自动启动的APP的完整路径名(注意这里的路径是在模拟器的Z盘)。还需要注意的地方就是把CPP自动生成的EXPORT_C CApaDataRecognizerType* CreateRecognizer() (在CPP文件的末尾,上面的代码已经直接删除掉了)这个函数注解掉,否则会报错。

    编译成功后你会在/z/system/recogs 目录下面发现MyRecognizer.def, MyRecognizer.lib MyRecognizer.mdl3个文件,其实有用的就是这个MDL文件。启动模拟器,你就会发现你想要自动启动的程序已经放在你眼前了,呵呵,成功了一半了。如果你想把它发布到真机(手机)上,就需要把MyRecognizer.mdl放到相应的recogs目录下面。

    总结一下,其整个运行过程就是 Symbian OS 内核启动-〉调用MDL-MDL启动线程(装载我们需要自启动的程序)-〉我们的程序运行。

    这里要顺便提两句,实现程序自启动还有其他方法,比如用现在比较流行的EZBOOT来实现,不过,我看过作者的介绍,其实也是用Recognizer来实现的,不过更加简便化而已。我想说的是,自己动手做一遍一定会受益匪浅。还要提醒大家,凡是都有两面性,特别是Recognizer,在里面写函数要特别小心,因为它是在OS加载的后要加载的核心MDL,一旦在里面实现的函数不够安全或者操作不当,很可能使手机当掉,最严重的是破坏FLASH。所以。。。不过上面的代码已经经过测试,可以安全使用。

    再次警告那些居心叵测的人,不要利用此程序发布恶意软件或者制作手机病毒。否则,带来的严重后果与本作者无关!

     需要自己编写的代码均已用斜体标注了;希望此作对大家有帮助。


    最新回复(0)