本文主要是介绍symbian 利用Recognizer编写自己的MDL实现开机自启动程序,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
声明:不得利用此程序制作手机病毒或恶意软件。否则,其带来的严重后果与本作者无关。
其实文章标题有些不准确。应该说利用Recognizer所提供的功能,可以使你的程序在Symbian系统启动后自动加载。Recognizer编译完成后其实是一个MDL,和DLL很类似。其Symbian OS 内部是怎样调用它的我现在不得而知,但是可以肯定的是当Symbian OS 内核启动完成后,一定会加载MDL无论它是否实现应该实现的功能。我看了Symbian OS SDK和Nokia论坛(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__ */
需要添加的LIB:apparc.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.mdl这3个文件,其实有用的就是这个MDL文件。启动模拟器,你就会发现你想要自动启动的程序已经放在你眼前了,呵呵,成功了一半了。如果你想把它发布到真机(手机)上,就需要把MyRecognizer.mdl放到相应的recogs目录下面。
总结一下,其整个运行过程就是 Symbian OS 内核启动-〉调用MDL-〉MDL启动线程(装载我们需要自启动的程序)-〉我们的程序运行。
这里要顺便提两句,实现程序自启动还有其他方法,比如用现在比较流行的EZBOOT来实现,不过,我看过作者的介绍,其实也是用Recognizer来实现的,不过更加简便化而已。我想说的是,自己动手做一遍一定会受益匪浅。还要提醒大家,凡是都有两面性,特别是Recognizer,在里面写函数要特别小心,因为它是在OS加载的后要加载的核心MDL,一旦在里面实现的函数不够安全或者操作不当,很可能使手机当掉,最严重的是破坏FLASH。所以。。。不过上面的代码已经经过测试,可以安全使用。
再次警告那些居心叵测的人,不要利用此程序发布恶意软件或者制作手机病毒。否则,带来的严重后果与本作者无关!
需要自己编写的代码均已用斜体标注了;希望此作对大家有帮助。
http://www.sf.org.cn/Article/symbiandev/200701/19987.html
///
不好意思,3版不用mdl,不过那个选项只有Microsoft VS的模板中才有~
没有其实也无所谓,很简单的处理:
1. 创建文件[uid3].rss //uid3改为你程序的Uid3
2. 编辑该文件,如下
#include <startupitem.rh>
RESOURCE STARTUP_ITEM_INFO dispatcher
{
executeable_name = "!://sys//bin//**app.exe";
recovery = EStartupItemExPolicyNone;
}
3. 将该资源放入mmp中编译
START RESOURCE [uid3].rss
TARGETPATH /private/101f875a/import
END
这篇关于symbian 利用Recognizer编写自己的MDL实现开机自启动程序的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!