PHP通过串口发短信

随技术进步,短信收发领域按时间先后产生了三种模式:BLOCK MODE,基于AT指令的TEXT MODE,基于AT指令的PDU MODE。其中,TEXT MODE比较简单,多款诺基亚手机均支持此款模式。西门子的手机大多数只支持PDU MODE。PDU 模式是收发短信的一种方法,短信正文经过十六进制编码后被传送。目前,PDU已取代BLOCK MODE。

SMS是由Etsi所制定的一个规范(GSM 03.40 和GSM 03.38)。当使用7-bits编码时,它可以发送最多160个字符;但用8-bit编码,最多可以发送140个字符,通常无法直接通过手机显示;还有用16-bit编码时,最多70个字符,被用来显示Unicode(UCS2)文本信息,可以被大多数的手机所显示。

     今天讨论的是PDU MODE,UCS2编码,也就是说,最多只能发送70个字符,不管英文还是中文。
     假设现在要发送如下信息:“你好”。在没有发送之前,要知道手机SIM卡所在地的短信中心号,例如移动的短信中心号:

      接收的手机号:13638197275
      杭州短信中心号:13800571500
      短信内容: 你好
    发送这条短信,要进行编码后手机才会执行,编码后会变成以下一串字符:
0891683180501705F011000D91683136187972F5000800044F60597D
   看不懂吧,从头到尾把这串编码解释一下:
       08 – 指的是短信中心号的长度,也就是指(91)+(683180501705F0)的长度除以2,即 08 =(2+14)/ 2
       91 – 指的是短信息中心号码类型。91是TON/NPI遵守International/E.164标准,指在号码前需加‘+’号;此外还有其它数值,但91最常用。
       683180501705F0  - 短信息中心号码。由于位置上略有处理,实际号码应为:8613800571500(字母F是补足偶数长度添加的字符)。
       11 - 文件头字节
       00 - 信息类型(TP-Message-Reference)
       0D - 被叫号码长度
       91 - 被叫号码类型

其实在实际处理中,我们通常把11000D91写死在程序中,因为在国内,这些数据都是不会改变的。

       683136187972F5 - 被叫号码,经过了位移处理,实际号码为“8613638197275”。

    上面的(00 )+(0D )+(91 )+(683136187972F5 ),构成了整个短信的第二部份目的地址(TP-Destination-Address)。

    继续...
    00 - 协议标识TP-PID,这里一般为00
    08 - 数据编码方案TP-DCS(TP-Data-Coding-Scheme),采用前面说的USC2(16bit)数据编码
    00 - 有效期TP-VP(TP-Valid-Period)
    04  - 长度TP-UDL(TP-User-Data-Length),也就是信息长度/2的十六进04
    4F60597D 这里就是短信内容了,实际内容为:“你好”

   根据以上情况,就可以写出短信编码的程序脚本了。

一、短信中心号码处理:
1、将短信息中心号码“+8613800571500”去掉+号,看长度是否为偶数,如果不是,最后添加F
=> “8613800571500F”
2、将奇数位和偶数位交换。
=> “683108501705F0″
3、将短信息中心号码前面加上字符91,91是国际化的意思
=> “91683108501705F0″
4、算出长度,结果除2,格式化成2位的16进制字符串,16 / 2 = 8 => “08″
=> “0891683108501705F0″

二、手机号码处理:
1、将手机号码+8613638197275去掉+号,看看长度是否为偶数,如果不是,最后添加F
=> “8613638197275F”
2、将手机号码奇数位和偶数位交换。
=> “683136187972F5″

三、短信息部分处理:
1、转字符串转换为Unicode代码,
“你好”的unicode代码 为4F60597D
2、将长度除2,保留两位16进制数,即 4F60597D = 8 / 2 => “04″,
=> “044F60597D″

四、组合
1、手机号码前加上字符串 11000D91(1100:固定,0D:手机号码的长度,不算+号,十六进制表示,91:发送
到手机为91,发送到小灵通为81),
即 11000D91 + 683136187972F5
=> 11000D91683136187972F5
2、手机号码后加上 000800 和刚才的短信息内容,000800也写死就可以了
即 11000D91683136187972F5 + 000800 + 044F60597D
=>  11000D91683136187972F5000800044F60597D
3、整条信息长度除以2,格式化成2位的十进制数
即 11000D91683136187972F5000800044F60597D => 38位 / 2 => 19

五、所以要发送的内容为
AT+CMGF=0 <回车> #此处为设定短信发送模式PDU
OK
AT+CMGS=19<回车>
> #输入短信内容编码

附加最终PHP代码:

    <?php  
    // Requirement dio, use cmd install: pecl install dio  
    set_time_limit(0);  
      
    // Windows use COM1:  
    $fd=dio_open('/dev/ttyS0', O_RDWR);  
    if(!$fd)  
    {  
        die("打开串口ttyS0失败");  
    }  
      
    // dio_tcsetattr() only Linux   
    // Windows 使用 exec('mode COM1: baud=9600 data=8 stop=1 parity=n xon=on');  
    dio_tcsetattr($fd, array(  
      'baud' => 9600,  
      'bits' => 8,  
      'stop'  => 1,  
      'parity' => 0  
    ));  
      
    //$ff=dio_stat($fd);  
    //print_r($ff);  
    //echo "GSM AT is start on ttyS0\n";  
      
    //短信中心号码  
    $smsc = "8613800571500";  
    $invert_smsc = invertNumbers($smsc); // 转换短信中心号码  
    $inter = chr(13); // 回车字符  
      
    $ctrlz = chr(26); // ctrl+z  
      
    // 发送信息  
    $text   
       = '你好';  
    $send_to = '8613638197275';  
    $pdu_phone  = hex2str(utf82unicode($text));  
    $pdu_phone  = sprintf("%02X", strlen($pdu_phone)/2) . $pdu_phone;  
    $pdu_phone  = '11000D91' . invertNumbers($send_to) . '000800' . $pdu_phone;  
    $atcmd      = 'AT+CMGF=0' . $inter;  
    @dio_write($fd, $atcmd);  
    $atcmd      = 'AT+CMGS=' . sprintf("%d", strlen($pdu_phone)/2) . $inter;  
    @dio_write($fd, $atcmd);  
    $pdu_addr   = '0891' . invertNumbers($smsc);  
    $pdu_all    = $pdu_addr . $pdu_phone . $ctrlz . $inter;  
    @dio_write($fd, $pdu_all);  
    dio_close($fd);  
      
    // 我的是utf-8编码   
    function utf82unicode($str)    
    {  
        return iconv("utf-8", "UCS-2BE", $str);  
    }  
       
    function hex2str($hexstring)    
    {  
        $str = '';  
        for($i = 0, $len = strlen($hexstring); $i < $len; $i++)  
        {  
            $str .= sprintf("%02X", ord(substr($hexstring, $i, 1)));  
        }  
        return $str;  
    }  
       
    function invertNumbers($msisdn)    
    {  
        $len = strlen($msisdn);  
        if ( 0 != fmod($len, 2) )   
        {  
            $msisdn .= "F";  
            $len = $len + 1;  
        }  
       
        for ($i=0; $i<$len; $i+=2)   
        {  
            $t = $msisdn[$i];  
            $msisdn[$i] = $msisdn[$i+1];  
            $msisdn[$i+1] = $t;  
        }  
        return $msisdn;  
    }  
       
    ?>  

 

 
附1 :各地移动短信中心号码
 
输入当地移动局的短消息号码:
+8613800xxx500("+"号必须输入),
其中xxx 为当地的电话区号。
--- 电话区号是三位的地区:
直接用电话区号替换xxx 即可。
例如:深圳电话区号是755,
移动的短消息中心号码是:+8613800755500
--- 电话区号是两位的地区:
请在区号后加"0"补足三位替换xxx。
例如:北京电话区号是10,
移动的短消息中心号码是:+8613800100500
目前联通165 网已在全国范围内开展业务。
在使用短信业务之前,需设置短信中心服务号码:
北京+8613010112500
上海+8613010314500
深圳+8613010888500
山东+8613010171500
江苏+8613010341500
浙江+8613010360500
福建+8613010380500
四川+8613010811500
重庆+8613010831500
海南+8613010501500
黑龙江+8613010980500
吉林+8613010911500
天津+8613010130500
河北+8613010180500
内蒙古+8613010950500
山西+8613010701500
安徽+8613010305500
新疆+8613010969500
青海+8613010776500
甘肃+8613010879500
宁夏+8613010796500
贵州+8613010788500
云南+8613010868500
湖南+8613010731500
湖北+8613010710500
广东+8613010200500
广西+8613010591500
河南+8613010761500
江西+8613010720500
辽宁+8613010240500
 
附2 :常用AT 指令:
AT+CSMS 选择短信息服务
AT+CPMS 选择短信息内存
AT+CMGF 选择短信息格式
AT+CSCA 短信息中心地址
AT+CNMI 显示新收到的短信息
AT+CMGR 读短信息
AT+CMGS 发送短信息
AT+CMGL 列出SIM 卡中短信息
AT+CMSS 从SIM 内存中发短信息
AT+CMGW 向SIM 内存中写入待发短信息
AT+CMGD 删除SIM 内存中的短信息
AT+CSCB 选择蜂窝广播信息
 
附3 :短信息的接收
接收短消息实质上就是从SIM 或缓存中读出信息。这主要利用AT+CMGR 和AT+CMG
L 两条指令来完成,由于无线模块不同的厂商对AT 指令集的解释代码和响应信息不一样,
所以首先要确认能否与MODEM 建立起通信,一般用AT 指令完成此确认;然后用AT+CM
GF 指令选定短消息的数据格式;在收到MODEM 的正确回答反以AT 指令完成读出功能。
一般用AT+CMGL 读取以前的信息,在收到MODEM 的RING(振铃)数据时,用AT+CM
GR 读取实时信息。以下是用H6221-W 的接收SMS 的一个实例,它说明了PDU 模式的应
用。
操作过程如下({}内为注释):
发送:AT
回答:OK {已建立联接}
发送:AT+CMGF=0 {选用PDU 格式}
回答:OK {允许选择PDU 格式}
发送:AT+CMGL=2 {列出已有的短信息}
回答:+CMGL:1,2,,24{1 表示信息个数,2 表示未发信息,24 表示信息总容量}
0D71683108370105F004000D81683179133208F10000026080410033802632184C
F682D
95E0DC2B36D3D170A0243106933D97A0243106933D97A02451068B1983492608
OK
以上这组PDU 格式的十六进制字符串,不但包含了短消息的内容,同时包含了发送者
的号码、短信息中心号码、短消息发送时间等。
下面对信息内容进行分析:
0D:短信息中心地址(号码)长度。
91:短信息中心号码类型,91 是TON/NPI。TON/NPI 遵守International/E.164 标准,
指在号码前需加'+'号;此外还可直有其他数值,但91 最常用。
683108370105F0:SMSC 短信息所使用的服务中心号码13807310500。它经过十六
进制以字节为单位的高低半字节换位处理,号码是奇数的添F,构成一个HEX 字节。
04:PDU 类型,文件头字节。
0B:主叫号码长度。
81:主叫号码类型。
3179133208F1:0A 主叫号码,也经过了处理,实际号码为13973123801。
00:PID,为协议标识。
00:DCS 短信息编码类型是GSM Default Alphabet,即由7 位ASCII 码移位组成8 位
十六进制码(octet),其方法见表2。
1sthex B0 A6 A5 A4 A3 A2 A1 A0
2ndhex C1 C0 B6 B5 B4 B3 B2 B1
3rdhex D2 D1 D0 C6 C5 C4 C3 C2
4thhex E3 E2 E1 E0 D6 D5 D4 D3
5thhex F4 F3 F2 F1 F0 E6 E5 E4
6thhex G5 G4 G3 G2 G1 G0 F6 F5
6thhex H6 H5 H4 H3 H2 H1 H0 G6
02608041003380:SCTS 短信息发送时间,02/06/08/14:00:33.08。
26:UDL 经处理后的8 位码(octet)短信息字节长度,它小于消息ASCII 码的长度。
32184CF682D95E30DC2B36D3D170A0243106933D97A0243106933D97A0245106
8B1983492608:UD 编码后的PDU 数据,短信息内容“2002/06/08/13:48ID102OKID103
OK ID201FAIL”。

posted on 2015-06-05 11:19  Fly_Wind  阅读(1789)  评论(2编辑  收藏  举报

导航