GSM/GPRS之二-短信pdu详细解析

目录:

一、概述

二、解析

三、编码与解码

1、 英文编码

2、 英文解码

3、 中文编码

4、 中文解码

附录


下续:GSM/GPRS之三-短信格式及编码


一、概述

对于短信的接收和发送,在传送时都会变成统一标准的16进制码,这里以接收为例,从RIL接收的是一个pud怎么转换成大家都可以认识的字符串了?

二、解析

RIL中收到new SMS,调用android.telephony.SmsMessage.newFromCMT() decode PDU,结果放入SmsMessage中。android.telephony.SmsMessage.newFromCMT() 根据电话的类型,用com.android.internal.telephony.gsm.SmsMessage或com.android.internal.telephony.cdma.SmsMessage的newFromCMT()真正的实现PDU decode。

解析pdu

对于RIL接收到到的pdu是一串16进制的数串,其含义会在下文中进行描述 

0891 683110802105F0 24 0D91 688102200982F6

00 08    21800381602423   044F60597D

1)短信中心内容解析 服务中心地址(Service Center Address,短信中心号码)

08 代表的是后面8个字节都是短信中心的内容

91代表的类型,91是国际类型,81或者A1表示是国内,如果是91就需要在后面的号码上‘+’

683110802105F0为短信中心号码,+8613010812500

------------------------------

2)pdu第一个字节解析 PDU类型(Protocol Data Unit Type)

24 代表的含义,其2进制表示0010 0100

接收的时候这8位分别表示以下含义:

BIT

7

6

5

4

3

2

1

0

参数

RP

UDHI

SRI

-

-

MMS

MTI

MTI

 发送的时候分别表示以下含义

发送的PDU,典型为11H: 

BIT

7

6

5

4

3

2

1

0

参数

RP

UDHI

SRR

VPF

VPF

RD

MTI

MTI

MTI 2bit:消息类型,00表示收,01表示发

MMS 1bit:短消息服务中心是否有更多短消息等待移动台。1有,0无。默认为1。

SRI 1bit:状态报告标示。0不需要状态返回到移动设备。1需要。默认为0。

UDHI 1bit:用户数据头标示。0用户数据没有头信息,1有。一般为0。

RP  1bit:是否有回复路径的标示。1有,0没有。一般为0。

VPF 2bit:有效期限格式。00 VP不存在; 10 VP区存在用一个字节表示,是相对值;

01保留;      11存在,半个字节表示,绝对值。

RD  1bit重复信元丢弃。0通知服务中心碰到同源、同目的地、同样的MR(短消息序号)的短消息接受;

1抛弃,此时将在短消息提交报告中返回一个适当的FCS值。

SRR 2bit:状态报告要求。

------------------------------

3)发送方地址

 0d 代表后面13,后面的字节数是[2+(Length+1)]/2=8,后面8个字节是发送方的地址;

 91 代表的是目的号码类型。

 688102200982F6+8618200290286,发送方的地址

 ------------------------------

4)TP-Protocol-Identifier(TP-PID) 发送方地址(Originator Address) 

 01TS 23.040 9.2.3.9

------------------------------

5)TP-Data-Coding-Scheme 协议标识(Protocol-Identifier)

 08:0000 1000

表示数据编码方法和消息类别。一般为00H默认7位编码,等级号0。UCS2编码0等级为08H,可以传输中文。

7

6

5

4

3    

 2

     1       

0              

编码组

保留

x

x

x

具体如下:

bit7-6

bit5

bit4

bit3-2

bit1-0

00

0:文本未压缩

0:表示bit1,

0是保留,没有消息类别;

0   0   默认字母表7bit

0    0     Class0

1:用GSM标准压缩

1:表示有

0   1    8bit数据

0    1     Class1

1   0    UCS2编码

1    0     Class2

1  1     保留

1    1     Class3

Bit1 Bit0  消息类别

00

0

0

10

00

Class0:短消息直接显示到用户终端

Class1:短消息存储在SIM卡上

Class2:短消息必须存储在SIM卡上,禁止直接传输到中断。

Class3:短消息存贮在用户设备上。

------------------------------

6)时间 服务中心的时间戳(Service Center Timestamp)

短信中心下发的时间戳,这个编码和长度固定

21800381602423

21: 年份,12

80:月,08

03:日,30

81:小时,18

60:分钟,06

12:秒,21

23:时区

------------------------------

7)UserDataHeader - UDH  用户数据长度(User Data Length(Amount of Characters))

04 4F60597D

04:UDL用户数据长度,包含用户数据和用户数据头的长度

1、 如果用户用默认7位编码。

1) 如果没有用户数据头,此数字标示7bit的字符个数。

2) 如果有用户数据头,此数字表示包括用户数据头(包含补丁在内)在内的7bit个数。即7bit个数加上头部长度在加1(补丁)

2、 如果用户用8位编码

表示用户数据区的字节数,有数据头信息,包括在内。

3、 如果为UCS2编码,则是用户数据区的字节数

注意:由于前面pdu的第一个字节里面已经标示是否有头,如果有头紧接的一个字节为头的长度,然后剩余的是用户数据的长度

------------------------------

8)UD 用户数据(User Data)(短消息内容)

4F60597D就是用户数据

三、编码与解码

1、 英文编码

缺省的GSM字符集为7位编码,ASCII码为8位编码,编码就是将8位ASCII编码转换为7位编码。
例如:1234 编码后得到31D98C06
2进制表示:
8位编码 00110001 00110010 00110011 00110100
7位编码 00110001 11011001 10001100 00000110 
通过例子可以看出,将ASCII8位编码的Bit8去掉,依次将下7位编码的后几位逐次移到前面,形成新的8位编码。
以下是C++Builder的实现代码:

String __stdcall EncodeEnglish(String InputStr)
{
  int n,len,cur;
  String tempstr,returnstr;
  unsigned char mid1[2],mid2[2];

  len=InputStr.Length();
  n=0;

  for(int i=1;i<=len;i++)
  {
    if (i<=len)
    {
      strcpy(mid1,InputStr.SubString(i,1).c_str());
      strcpy(mid2,InputStr.SubString(i+1,1).c_str());
      cur=(mid1[0]>>n)|((mid2[0]<<(7-n))& 0xff);
    }
    else
    {
      strcpy(mid1,InputStr.SubString(i,1).c_str());
      cur=(mid1[0]>>n)& 0x7f;
    }

    FmtStr(tempstr,%2.2X,ARRAYOFCONST((cur)));
    returnstr=returnstr+tempstr;
    n=(n+1)%7;

    if (n==0)
      i++;
  }
  return returnstr;
}

2、 英文解码

简单地说就是将7位字符编码转换为8为字符编码。

以下是C++Builder的实现代码:

int ReturnHex(int Value)
{
  switch (Value)
  {
    case 0:Value=0x7f;break;
    case 1:Value=0x3f;break;
    case 2:Value=0x1f;break;
    case 3:Value=0x0f;break;
    case 4:Value=0x07;break;
    case 5:Value=0x03;break;
    case 6:Value=0x01;break;
    case 7:Value=0x00;break;
  }
  return Value;
}

String __stdcall DecodeEnglish (String InputStr)
{
  unsigned char InStr[300];
  char OutStr[300];
  String str;
  int j=0,i=0;
  int Point=0;
  int temp;

  memset(InStr,0,301);
  memset(OutStr,0,301);

  for(int i=0;i小于InputStr.Length();i等于i+2)
  {
    str=0x+InputStr.SubString(i+1,2);
    InStr[i/2]=StrToInt(str);
  }

  while(j<=InputStr.Length()/2)
  {
    if(Point==0)
      OutStr[i]=InStr[j]&ReturnHex(Point);
    else
      OutStr[i]=((InStr[j]&ReturnHex(Point))<<point)|(instr[j-1]>>(8-Point));

    if((Point%7==0)&&(Point!=0))
      Point=0;
    else
      Point=Point+1;

    i++;
    j=i-(i/8);
  }

  OutStr[12]=((InStr[12]&0x07)<<5)|(InStr[11]>>(8-5));
  return AnsiString(OutStr);
}

3、 中文编码

中文编码较为简单,就是将GB2312的中文编码转换为代码页为CP936的Unicode编码即可。

以下是C++Builder的实现代码

String EncodeChinese(String InputStr)
{
  int cur;
  String tempstr,returnstr;
  WideString ws;
  wchar_t mid[2];
  ws=WideString(InputStr);

  for(int i=1;i<=ws.Length();i++)
  {
    wcscpy(mid,ws.SubString(i,1).c_bstr());
    cur=mid[0];
    FmtStr(tempstr,%4.4X,ARRAYOFCONST((cur)));
    returnstr=returnstr+tempstr;
  }
  return returnstr;
}

4、 中文解码

将代码页为CP936的Unicode编码转换为GB2312的中文编码即可。

以下是C++Builder的实现代码

String DecodeChinese(String InputStr)
{
  wchar_t Buf[300];
  for(int i=0;i
  {
    Buf[i/4]=StrToInt(0x+InputStr.SubString(i+1,4));
  }
  Buf[InputStr.Length()/4]=0;
  return WideCharToString(Buf);
}

附录

GPRS用AT命令发送中文短信(TEXT模式到PDU模式的转换):GPRS用AT命令发送中文短信


最要紧的是果敢的迈出第一步,对与错先都不管。觉得不错,动动发财的小手点个赞哦!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

爱上电路设计

你的鼓励是我创作最大的动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值