Benjamin

静以修身,俭以养德,非澹薄无以明志,非宁静无以致远。
随笔 - 397, 文章 - 0, 评论 - 196, 引用 - 0
数据加载中……

WM5.0下如何获取基站信息(LAC、Cellid)

在WM5.0下获得CELLID、LAC的途径有两条:利用串口发送AT指令或是利用RIL来获取。RIL(Radio Interface Layer)是微软自己开发的一个库,它的程序有固有的特点,在获取CELLID上,它其实是对第一种方法的封装,两者本质是一样的。但要注意是:串口一旦打开,就难以关闭,除非重启机器(可能涉及到底层的中断),另外并不是所有的设备都可以取到CELLID。
下面的这段代码就是利用COM口来获取CELLID,并不保证所有的设备都支持。

typedef struct
{
 char c[12];
 char    iLac[4];
 char s[4];
 char iId[4];
} TCREG_DATA;

void Get_Cellid(void)
{
 char m_sTemp[12] = {0};
 strcat(m_sTemp,"COM");
 for(int i = 9; i > 0; -- i)
 {
  char ch1;
  _itoa(i,&ch1,10);
  strcat(m_sTemp,&ch1);
  strcat(m_sTemp,":");

  TCREG_DATA* pData = (TCREG_DATA*)GetCREG(m_sTemp);
  if(!pData)
   continue;

  char szNum1[8] = {0};
  char szNum2[8] = {0};

  strcpy(szNum1,pData->iLac);
  strcpy(szNum2,pData->iId);

  int iLac = (int)strtol(szNum1,NULL,16);
  int iId  = (int)strtol(szNum2,NULL,16);

  if (iLac && iId)
  {
   sprintf(m_sCell.LAC,"%06d",iLac);
   sprintf(m_sCell.ID,"%06d",iId);
   break;
  }
 }
}
char* GetCREG( char * comPort )
{
 HANDLE hCom;
 int bufpos;
 DCB dcb;
 COMMTIMEOUTS to;
 DWORD nWritten;
 DWORD event;
 DWORD nRead;
 static char outbuf[20], buf[256];

 BYTE comdevcmd[2]= {0x84, 0x00};
 WCHAR m_sCom[12] = {0};
 mbstowcs(m_sCom,comPort,strlen(comPort));

 hCom= CreateFile( m_sCom ,GENERIC_READ|GENERIC_WRITE,0,0,OPEN_EXISTING,0,0);

 if (hCom==NULL || hCom==INVALID_HANDLE_VALUE)
 {
  hCom= NULL;
  return NULL;
 }

 if (!GetCommState(hCom, &dcb))
 {
  return "ERROR:GetCommState Failed";
 }

 dcb.BaudRate= CBR_115200;
 dcb.ByteSize= 8;
 dcb.fParity= false;
 dcb.StopBits= ONESTOPBIT;

 if (!SetCommState(hCom, &dcb))
 {
  return "ERROR:SetCommState Failed";
 }

 EscapeCommFunction(hCom, SETDTR);
 EscapeCommFunction(hCom, SETRTS);
 GetCommTimeouts(hCom, &to);
 //if (!EscapeCommFunction(hCom, SETDTR))
 //{
 //  return "-4";
 //}
 //if (!EscapeCommFunction(hCom, SETRTS))
 //{
 //  return "-5";
 //}   
 //if (!GetCommTimeouts(hCom, &to))
 //{
 //  return "-6";
 //}

 to.ReadIntervalTimeout= 0;
 to.ReadTotalTimeoutConstant= 200;
 to.ReadTotalTimeoutMultiplier= 0;
 to.WriteTotalTimeoutConstant= 20000;
 to.WriteTotalTimeoutMultiplier= 0;

 SetCommTimeouts(hCom, &to);
 //if (!SetCommTimeouts(hCom, &to))
 //{
 //  return "-7";
 //}

 if (!SetCommMask(hCom, EV_RXCHAR))
 {
  return "-8";
 }

 DWORD rildevresult=0,nReturned=0;


 if (!DeviceIoControl (hCom,0xAAAA5679L, comdevcmd, sizeof(comdevcmd),0,0,0,0))
 {
  return "-9";
 }

 bufpos = 0;

 strcpy(outbuf,"AT+creg=2\r");

 if (!WriteFile(hCom, outbuf, 10, &nWritten, NULL))   
 {
  return "-10";
 }

 if (nWritten != 10)
 {
  return "-11";
 }

 if (!WaitCommEvent(hCom, &event, NULL))
 {
  return "-12";
 }

 while(1)
 {
  if (!ReadFile(hCom, buf+bufpos, 256 - bufpos, &nRead, NULL))
  {
   return "-13";
  }

  if (nRead == 0)
   break;

  bufpos += nRead;


  if (bufpos >= 256)
   break;
 }

 strcpy(outbuf,"AT+creg?\r");

 if (!WriteFile(hCom, outbuf, 9, &nWritten, NULL))   
 {
  return "-14";
 }

 if (nWritten != 9)
 {
  return "-15";
 }

 if (!WaitCommEvent(hCom, &event, NULL))
 {
  return "-16";
 }

 while(1)
 {
  if (!ReadFile(hCom, buf+bufpos, 256 - bufpos, &nRead, NULL))
  {
   return "-17";
  }

  if (nRead == 0)
   break;

  bufpos += nRead;

  if (bufpos >= 256)
   break;
 }

 puts(buf);

 rildevresult = 0; 

 if (!EscapeCommFunction(hCom, CLRDTR))
 {
  return "-4";
 }

 if (hCom!=NULL)
 {
  CloseHandle(hCom);
  hCom= NULL;
 }

 char* cregResponse = strpbrk( buf, "CREG\0" );

 return cregResponse;
}

void Get_Cellid(void)
{
 char m_sTemp[12] = {0};
 strcat(m_sTemp,"COM");
 for(int i = 9; i > 0; -- i)
 {
  char ch1;
  _itoa(i,&ch1,10);
  strcat(m_sTemp,&ch1);
  strcat(m_sTemp,":");

  TCREG_DATA* pData = (TCREG_DATA*)GetCREG(m_sTemp);
  if(!pData)
   continue;

  char szNum1[8] = {0};
  char szNum2[8] = {0};

  strcpy(szNum1,pData->iLac);
  strcpy(szNum2,pData->iId);

  int iLac = (int)strtol(szNum1,NULL,16);
  int iId  = (int)strtol(szNum2,NULL,16);

  if (iLac && iId)
  {
   sprintf(m_sCell.LAC,"%06d",iLac);
   sprintf(m_sCell.ID,"%06d",iId);
   break;
  }
 }
}

posted on 2009-02-04 22:48 Benjamin 阅读(3781) 评论(5)  编辑 收藏 引用 所属分类: PDA/PPC开发

评论

# re: WM5.0下如何获取基站信息(LAC、Cellid)  回复  更多评论   

你好,我不太懂手机内部AT发送与接收流程。两个模块间的虚拟串口如何操作,如果获取后对串口操作会有什么影响吗?在windows mobile中不是一启动ril就占用串口吗?
2009-08-11 09:58 | dane

# re: WM5.0下如何获取基站信息(LAC、Cellid)  回复  更多评论   

获取后串口的关闭是个难题,即使用了关闭了句柄也无济于事,好像必须要重启机器。如果获取失败,有的机器会死机,所以建议用RIL来获取,RIL封装的比较好,其底层实现和上述代码机制是一样。
另外在每种型号的机器上,能使用的串口也不一样,在注册表中有些串口是不能用的,系统分配好的,但是每种机器都不同。
模拟串口用软件可以实现,具体的可以在网上搜搜。
2009-08-12 21:29 | Benjamin

# re: WM5.0下如何获取基站信息(LAC、Cellid)  回复  更多评论   

@Benjamin
你写的这个程序是通过电脑发送到手机的吗?
我想知道如何获取发送AT指令的串口;
我通过程序获取的都是我配置的一些串口;
我使用的是仿真器,和真机在获取串口时用什么区别吗?
那你知道如何获取TXpower,相邻小区的信息,c1,c2等信息吗?
2009-08-24 13:39 | dane

# re: WM5.0下如何获取基站信息(LAC、Cellid)  回复  更多评论   

程序是直接运行在手机上,这个串口当时我是一个个的试的,在注册表中没找到。模拟器上我没试过,应该获取不了吧。
至于其他的信息,建议你在网上搜搜AT指令或微软的RIL库。
2009-08-26 21:30 | Benjamin

# re: WM5.0下如何获取基站信息(LAC、Cellid)  回复  更多评论   

你是用的是WM5.0?那不是当启动启动时device.exe加载了RIL Driver占用了能够发送AT指令的串口。我试遍了所有的串口就是不能发送AT指令,有的不能打开,有的发送过去没有反应。。不知道问什么。网上的资料就是WM手机启动时就占用了与通信模块发送AT的串口所以不能做到。你是怎么解释的。
2009-10-21 14:08 | dane

只有注册用户登录后才能发表评论。
网站导航: 博客园   IT新闻   BlogJava   博问   Chat2DB   管理