Answer

专注于Mobile,WinCE
  首页  :: 新随笔  :: 联系 :: 管理

[转]WinCE下消息队列用法

Posted on 2010-05-06 15:14  answer  阅读(726)  评论(0编辑  收藏  举报
驱动和AP之间互相通信有很多中方法,比如自定义Event然后AP通过等待Event的方式来和driver同步。但是很多标准的事件,比如电源状况的改 变,SD卡插入等等通用的消息在CE的内部是早有预留的。微软透过消息队列的形式来告诉需要获知相关消息的AP来做处理,我们仅仅需要事先去了解系统有哪 些消息队列,然后在AP中写代码去捕捉就好了,本文就以获知电池电量改变为例,讲讲消息队列的用法。

     在WinCE的项目开发过程中经常要编写AP来获取电池的电量和电源的来源等信息,由于WinCE底层的电池驱动一般以查询的方式得到电池的状态然后更新 到一个结构体中,AP可以调用GetSystemPowerStatusEx2来得到这个结构体的数值,为了实时的更新电池的信息AP必须频繁的调用函数 去得到数据更新。

     其实WinCE的电源管理中已经集成了一种notify机制,会在电池信息发生变化时发出提醒。

     RequestPowerNotifications函数可以被AP用来请求接收这种提醒服务。

     AP在调用这个API之前必须创建一个消息队列,可以用CreateMsgQueue来实现。

     接受提醒的方式是使用WaitForSingleObject来实现,该函数会一直等待直到收到电源管理发来的提醒,然后AP可以去读取消息队列中的数据来判定具体电源系统发生了哪些变化,然后做相关的事情比如更新UI的显示等。

参考源代码:
//#################################################################


#include

#define  QUEUE_ENTRIES    3   

#define  MAX_NAMELEN      200   

#define  QUEUE_SIZE       (QUEUE_ENTRIES  *  (sizeof(POWER_BROADCAST)  +  MAX_NAMELEN))   

HANDLE hMsgQ;

DWORD WINAPI PowerChangeListen(void * temp_p)

{

UCHAR  buf[QUEUE_SIZE];   

unsigned long nRead = 0, flags = 0, res = 0;

while(1)

{

  DWORD dwRes = WaitForSingleObject(hMsgQ,INFINITE);

  if(dwRes==WAIT_OBJECT_0)

  {
   memset(&buf,  0,  QUEUE_SIZE);   

   if (ReadMsgQueue(hMsgQ, &buf, QUEUE_SIZE, &nRead, INFINITE, &flags))

   {

    PPOWER_BROADCAST  pB  =  (PPOWER_BROADCAST)&buf;

    PPOWER_BROADCAST_POWER_INFO  ppbpi  =  (PPOWER_BROADCAST_POWER_INFO)  pB->SystemPowerState;  

    if(pB->Message==PBT_POWERINFOCHANGE)

    {

     //在这里处理一些电池信息相关数据改变的事情

     // MessageBox(NULL,L"Battery info change",NULL,NULL);

     NKDbgPrintfW(L"[Fred]Battery info change BatteryLifePercent=%d\r\n",ppbpi->bBatteryLifePercent);

    }



    if(pB->Message==PBT_POWERSTATUSCHANGE)

    {

     //在这里处理一些电源输入状态改变 (AC/Battery)的事情

      //MessageBox(NULL,L"Power input change",NULL,NULL);

     NKDbgPrintfW(L"[Fred]Power input change ACIN=%d\r\n",ppbpi->bACLineStatus);

    }

  }

}

}
}

void Init_PowerNotify()

{

NKDbgPrintfW(L"[Fred]Init_PowerNotify++\r\n");

MSGQUEUEOPTIONS  options  =  {0};

DWORD dwErr;

          options.dwSize  =  sizeof(MSGQUEUEOPTIONS);   

          options.dwFlags  =  0;   

          options.dwMaxMessages  =  QUEUE_ENTRIES;   

          options.cbMaxMessage  =  sizeof(POWER_BROADCAST)  +  MAX_NAMELEN;   

          options.bReadAccess  =  TRUE;  

           hMsgQ  =  CreateMsgQueue(NULL,  &options);  

if(!hMsgQ)

{

  dwErr=GetLastError();

  NKDbgPrintfW(L"[Fred]CreateMsgQueue failed\r\n");

     RETAILMSG(1,  (TEXT("[Fred]CreateMessageQueue  ERROR:%d\n"),  dwErr));   

  return;   

}



HANDLE hNotifications  =  RequestPowerNotifications(hMsgQ, POWER_NOTIFY_ALL);  //  Flags   

          if  (!hNotifications)  {   

                  dwErr  =  GetLastError();   

                  RETAILMSG(1,  (TEXT("[Fred]RequestPowerNotifications  ERROR:%d\n"),  dwErr));   

                  StopPowerNotifications(hMsgQ);

      return;

          }   

CreateThread(NULL, 0, PowerChangeListen, NULL, 0, NULL);

NKDbgPrintfW(L"[Fred]Init_PowerNotify--\r\n");

}

//###################################################################################

复制代码AP可以把上面的代码全部复制到自己的源码中,然后在初始化的时候调用一次Init_PowerNotify,之后就可以等待消息的发生(中文注释部分)

 

 来自:

http://blog.csdn.net/armeasy/archive/2009/10/09/4647131.aspx