电信CDMA PDU短信号码字段编码转换的python实现

参考链接:CDMA(3GPP2)SMSPDU全解析-爱码网

CDMA SMS pdu解码_zx249388847的博客-CSDN博客_cdma sms
cdma pdu解析终极文档_月之海的博客-CSDN博客
CDMA PDU 编解码补充解析_zx249388847的博客-CSDN博客

CDMA – lytsing's Blog

一、PDU串解析

        CDMA的pdu格式与GSM的相差很多,不能直接用肉眼看出来。第一条内容比较长,就拿第二条pdu串来分析吧。

1.首先将PDU串打成PDU包将PDU串相邻的两个ascii字符拼凑成一个8bit数据

如下:
00 00 02 10 02 02 07 02 c5 4c e2 25 a8 a8 06 01 4c 08 4d 00 03 10 01 f8 01 3e 20 f0 01 90 01 78 01 90 01 62 df ca 70 04 b1 ac b1 ab b4 23 96 14 c6 70 01 62 96 3b 2b 12 b9 82 7a e3 10 c0 01 72 9f 54 4c 0b 10 8b b4 23 91 8a 75 d0 01 63 31 7a 70 02 9f 52 e0 7c f0 f8 03 06 08 12 29 19 26 16

消息传送类型: 0x00     point to point message(表示点对点消息)

下面的内容为短消息的各个字段 每个字段分为三个部分: 字段类型(ID)(8bit)、字段长度(Length)(8bit)和字段内容.

第一个字段: 00 02 10 02
        0x00, 表示uTeleserviceID字段
        0x02, 字段长度,该长度必须为2,否则为错误的pdu信息

        字段内容为:0x1002,十进制是4098,

第二个字段: 02 07 02 c5 4c e2 25 a8 a8
        0x02, SMS_TL_ORIG_ADDR 表示 (短信发送地址)
        0x07, 字段长度为7


字段内容: 02 c5 4c e2 25 a8 a8
只看前面几个 02 c5 4c e2的:
        0000 0010 1100 0101 0100 1100 1110 0010 0010
取第一个bit 0 表示 RIL_CDMA_SMS_DIGIT_MODE_4_BIT 是4bit压缩
第二个bit 0 表示 RIL_CDMA_SMS_NUMBER_MODE_NOT_DATA_NETWORK
下来8个bit 是 00 0010 11 = 11,表示号码长度 为11
由于是4bit压缩 ,后面44个bit(4*11)表示号码,解析出来是15338896020

第三个字段: 06 01 4c
        表示SMS_TL_BEARER_RPLY_OPT

第四个字段:08 4d 00 03 10 01 f8 01 3e 20 f0 01 90 01 78 01 90 01 62 df ca 70 04 b1 ac b1 ab b4 23 96 14 c6 70 01 62 96 3b 2b 12 b9 82 7a e3 10 c0 01 72 9f 54 4c 0b 10 8b b4 23 91 8a 75 d0 01 63 31 7a 70 02 9f 52 e0 7c f0 f8 03 06 08 12 29 19 26 16

0x08, 表示SMS_TL_BEARER_DATA字段(短信内容)
0x4d,字段长度为77

这个字段也分成各个子字段:

第一子字段: 00 03 10 01 f8
0x00 Mesage Id
0x03 内容长度
10 01 f8  === 0001 0000 0000 0001 1111 1000
0001 表示 DELIVER 短信
0000 0000 0001 1111表示 message id.
紧接后面的 1, 表示HEADER_IND

第二个子字段:01 3e 20 f0 01 90 01 78 01 90 01 62 df ca 70 04 b1 ac b1 ab b4 23 96 14 c6 70 01 62 96 3b 2b 12 b9 82 7a e3 10 c0 01 72 9f 54 4c 0b 10 8b b4 23 91 8a 75 d0 01 63 31 7a 70 02 9f 52 e0 7c f0 f8
0x01, 字段类型
0x3e(62),字段长度
20 f0 01 === 0010 0000 1111 0000 0000 0001
0x20 的前5个bit为00100, 为0x04, 表示短信编码方式为RIL_CDMA_SMS_ENCODING_UNICODE (UNICODE)
0x20 的后3个bit, 0xf0的前5个bit,为 000 1111 0, 即0x1e (30),表示有30个UNICODE 字符。0xf0的后3个bit,0x01的8个bit,再加上0x90的前5个bit,

0000 0000 0011 0010 拼成一个16位数是 0x0032 表示字符 :2,在vim下了解一个字符的16进制码很简单,光标在该字符,按ga,底端显示结果如下:

<2>  50,  Hex 32,  Octal 062                                  2,1           All

依次下去,内容就是上面运行./test_pdu_decode 的结果: 2/2,对一阵阵的狂风,勇敢地作战.只要我的爱人,是一只小鸟

第三个子字段: 03 06 08 12 29 19 26 16

这是一个时间戳字段: 08年12月29号19时26分16秒(短消息发送时间)

二、地址参数字段

040702C4CC48489858       ========== 》Address Parameters(地址参数字段):

此字段为可变长度

参数ID号1个字节:04表示目的地址(接收方号码) 02表示发起方地址(发送方号码)

参数长度1个字节:07表示后面的数据占7个字节

号码编码模式1bit:0表示使用4bit的未知编码方案的DTMF编码 1表示使用8bit的指定编码方案的8bit编码

号码地址模式1bit:0表示使用ANSI TI.607定义的地址  1表示使用数据网络地址格式

号码类型0或者3个bit:如果号码编码模式为0,此字段忽略

号码编码方案0或者4个bit:如果号码编码模式为0或者号码地址模式为1,此字段忽略

号码个数8bit:指示号码字符个数

号码数字或者字符的编码:4bit或者8bit编码

填充字段0到7bit:保证地址参数字段为整数个字节,填充为0

将02C4CC48489858转化为2进制数据:

0000 0010 1100 0100 1100 1100 0100 1000 0100 1000 1001 1000 0101 1000

再按字段组合:

0 0 00001011 0001 0011 0011 0001 0010 0001 0010 0010 0110 0001 0110 00

从上面的字段组合可以看出:

号码编码模式取0表示使用4bit编码

号码地址模式取0表示使用ANSI TI.607定义的地址

由于号码编码模式取0没有号码类型和号码编码方案

号码个数为11个,号码为13312122616,填充字段为00

三、python转换实现地址参数字段

 进制转换地址参考:二进制转换器 - 在线进制转换 - 木鱼查询

首先需要将手机号码由十六进制转换成二进制:

b = bin(int(number_change, 16))
print('十六进制转二进制后:{}'.format(b[2:]))

但有个问题,那就是关于号码中的0字段,需要转换成A,即

number_change = number.replace('0', 'A')
print(f'号码转换前:{number},转换后:{number_change}')

当前编码的是发送的短信,可以参考前面的内容,拼接语句:

bin_sum = '100000001110000001011000' + b[2:] + '00'
print(f'转二进制后完整字段内容:{bin_sum}')

之后进行十六进制转换:

hex_s = hex(int(bin_sum, 2))[2:]
print(f'转十六进制后内容输出:{hex_s}')

当前040702C4CC48489858仅进行40702C4CC48489858转换,前面的0可后面直接添加

def numbertopdu(number):
    number_change = number.replace('0', 'A')
    print(f'号码转换前:{number},转换后:{number_change}')
    b = bin(int(number_change, 16))
    print('十六进制转二进制后:{}'.format(b[2:]))
    bin_sum = '100000001110000001011000' + b[2:] + '00'
    print(f'转二进制后完整字段内容:{bin_sum}')
    hex_s = hex(int(bin_sum, 2))[2:]
    print(f'转十六进制后内容输出:{hex_s}')
    return hex_s

phonenumber = '13312122616'
numbertopdu(phonenumber)

运行结果如下:

 

 比较

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值