找回密码
 加入我们

QQ登录

只需一步,快速开始

搜索
查看: 1531|回复: 3

[CDMA] 水货CDMA写号机不能接收彩信的原因及自动收发彩信解决方案

[复制链接]
发表于 2012-10-9 17:55 | 显示全部楼层 |阅读模式
本帖最后由 nickjun 于 2012-10-9 18:00 编辑 $ j8 x! X# C& Y1 t3 Y4 Y

3 z; w4 c1 _& P7 b. L- ]文章转自MIUI论坛,目前webos已经开源,分析一下代码应该可以找到相关设置,以达到彻底解决CDMA palm pre彩信的问题。
8 L% r% N# R/ D& w) X0 \1 `http://www.miui.com/thread-378600-1-1.html7 j: h( G6 I2 A; m) M! F. t
经过N天的努力和尝试,终于解决了i500 CDMA水货机美版MIUI的彩信接收问题。目前测试的结果,有很多朋友已经可以完美接收彩信。相关补丁已发布在diypda的i500版块,使用i500的朋友可前去围观。
' f" V7 M4 L# g$ J- X* h7 h+ M' V
7 i; V* w" n2 t6 u" M# U! k3 d有图有真相。。。。。。( |9 @& i' X2 x+ J4 I0 y& R& l

% o' V- z: F+ L* _! I
2 Q% q- z: k8 f4 s# u( f: S( x
# S+ e' q) S1 B& s7 e本机环境:三星i500 showcase写号机,美版MIUI 1.12.9 for i500 fascinate。
# ?: }; c  t5 o' ~/ ]# B
( g5 A' j6 O- x" K3 {虽然我只是针对i500定制,但从理论上来说,这个解决方案对所有的水货机均应适用。
( k# w- T, [+ ? 7 X" c5 Z$ L* a1 h& [
$ j7 G4 u2 w' E2 @4 p& h( B) r
众所周知,现在大部分的好手机都是GSM/WCDMA制式的,CDMA/EVDO在国内基本上没有什么好看兼好用的机器。偶尔有些好一点的WCDMA机型,让电信的大爷们一“定制”,立刻变得歪瓜裂枣巨丑无比,让我深深怀疑自己的审美观价值观人生观是否正确并代表先进的生产力方向。
% n5 Q" b* v% c0 \( ~
9 F0 A/ B5 D. b3 N3 p众里寻他千google,终于找到了一款外形和配置还算说得过去的机型。三星的i500。但很遗憾的是,这款机型国内并没有引进,某宝上买到的,都是美版的机子。写号倒不是很麻烦,可水货机始终不能接收彩信,却是一直困扰很多水货机用户的大问题。
9 e0 |( l) ]( b! L( O; t
5 D6 i) ~$ H5 k5 [+ K  e虽然彩信是上个世纪的东西了,但日常生活中还是用得比较多的。比如我,虽然一般是不发彩信的,但经常会收到电信或其它商家发送的彩信,有些网上的信息比如优惠券,也是以彩信的形式发送的。作为一项手机功能,用不着的时候无所谓,但一旦用着了却不能用,就很让人郁闷了。
! O- W  C. |" G2 b! G5 n/ R
  N6 `+ t8 C4 C我一直坚持的生活原则就是,宁可千日不用,但不能一日没有。。。。。。
/ T+ Y- }2 c" J/ y/ T% ^9 T& b/ [ + _" [9 p6 N! j
彩信的发送就不用说了,设置正确的apn,即可发送彩信,这个比较简单。下面主要说一下彩信的接收问题。, o7 {* \! Y$ `  I+ t' d5 X
. z  n0 E4 `! _; j0 \4 ?9 E$ v
1、不能接收彩信的原因
/ c" @& D8 E$ P! l; j" r 9 X( l& Y' Z, w, J3 l& D
在大量搜索网上资料的基础上,经过几天的测试和分析,找到了不能接收彩信的初步原因。1 ]) F4 O, e* R* P

7 y% Y+ _1 r  @' l' l彩信,按照电信规范,是分两步发送的。
1 l# j) H/ d. r' z 第一步,发送WAP-PUSH格式的短信通知,该短信通知含有一个url,该url指向彩信内容的存储地址。
( @& n5 q, a- \$ P. V1 ^ 第二步,手机解析出这个url,向url发送获取请求,下载彩信内容到本地,完成彩信的接收。9 V7 S5 t! d* _* E- T
: X, Q( X( Q. p( |0 j" g
向手机发送彩信之后,手机上没有任何的反映,说明第一步的彩信通知都没有接收到。
; k  P3 f! h) c# [8 U; v
# ]3 u6 w6 w3 o% v8 z5 {我写了一个Receiver,拦截系统中广播的android.provider.Telephony.WAP_PUSH_RECEIVED消息,结果没有收到这个广播。证明彩信通知并没有上传到app应用层,那么有可能在framework框架层,彩信通知就被拒绝了。3 b7 A: W/ L; Z* m$ l1 {$ d5 S
. ]9 D. N% `  v1 L# W) c/ G( P
在Receiver中添加android.provider.Telephony.SMS_REJECTED,收到了系统广播的消息。说明彩信通知确实是被framework层给reject了。9 s% Z- _' \) `1 Q- J- H

3 J7 e% W0 C5 b取出/system/framework/framework.jar,解压出classes.dex,对dex文件进行baksmali,再对照android系统源码进行流程分析(虽然只是简单的一句话,但确实够我忙活了一段时间。。。。。。),最终确定,问题在com.android.internal.telephony.cdma.CdmaSMSDispatcher.smali中。1 }1 K$ B, a+ ]# ?
5 W" r* l0 T. D* O+ r8 g
对照android源码,找出问题的关键代码段:
  1. if (SmsEnvelope.TELESERVICE_WAP == teleService) {8 M5 H. A3 e; A
  2.     return processCdmaWapPdu(sms.getUserData(), sms.messageRef,
    ! x. W7 k+ O' j0 }  l$ l+ U: X% A
  3.             sms.getOriginatingAddress());, B9 v- D! q0 j  ]
  4. }
    5 K4 n7 p$ [' J
  5. 2 `  ?2 K7 d& ]  S7 y  [
复制代码
查找smali中对应的TELESERVICE_WAP,发现其设定的值是0x1004,换算成10进制就是4100。: q  y1 r- F) `4 ]( H
( {( s, a+ x9 ^' O1 P
修改CdmaSMSDispatcher.smali,使之在返回sms rejected的原因时,填入彩信通知的TeleServiceId,发现电信的TeleServiceId确实是65002,中国电信CDMA终端规范也证实了这个判断。
& M: H& S+ y+ G+ N% g # E" V* T3 N$ x
那么原因找到了,由于TeleServiceId的不同,导致手机自带的美版ROM无法识别65002的电信Id,所以拒绝接收彩信通知。5 H8 l9 F& n; S7 ~( x3 t( k

9 q: F& p; U5 l' Z
9 w' _: Q& y" k# s2、彩信接收解决方案
9 O& i/ G. A0 ^; [& P5 ?! @9 _( o 4 o4 S: ~  q6 p1 S5 F& u7 V+ C! ?
找出原因来了,当时我就想,这应该就比较简单了吧。。。。。。: x6 e: }# U4 z4 H3 C% h% c
5 _1 R  \- D# ^" p2 i8 H
2.1 修改TeleServiceId
# T* `$ W1 _; D " S+ y1 E: S* t" g. x
定位CdmaSMSDispatcher.smali中判断TeleServiceId的语句:
  1. const v10, 0x1004
    + Z* ^) ]8 S5 a" f8 H# H5 a
复制代码
把const/16 0x1004(4100)修改成const 0xfdea(65002),使之识别出这是一个WAP-PUSH,进而转入processCdmaWapPdu流程,处理彩信通知的pdu。$ N' Z+ b; `% \4 q; ^5 ?

. b9 `# @2 V% K. _编译所有的smali文件,更新classes.dex,替换本机中的framework.jar,重启手机。
# w1 A( X" Z+ D4 f1 ?
, ^9 O! Z5 c4 A# R怀着激动的心情发送彩信,一直在等。。。。。。终于等来了。。。。。。“您的手机终端不支持接收彩信,请到mmbox.vnet.cn中提取”。。。。。。- }3 _6 f% z7 a. u( `5 r
/ c; N. Z& b5 e$ t2 ~, b/ g
2.2 修改pdu处理流程
  o5 m0 P7 s1 `# ]: Z3 w ( v( b: S( }0 t$ Z" B
我很郁闷,明明把TeleServiceId改过来了,为什么还是收不到。
! G1 H2 y6 k- U. _& a  m
3 u# t7 r0 E' q. c% G问题一定出在processCdmaWapPdu这个方法里。又经过了一段时间的调试和分析(又是简单的一句话,背后是N个小时的连续奋战和N元钱的彩信发送费用。。。。。。),终于发现,pdu解析有问题。0 n1 I8 E) ]$ o# P
- v+ _5 C% C2 J. M- U
processCdmaWapPdu传入的第一个参数,sms.getUserData(),获取的是pdu中的包括msg identifier在内的完整userData,而processCdmaWapPdu中的代码,对传入的参数是按照CHARi来进行处理的,所以导致获取的totalSegments和segment完全是错的,系统认为收到的彩信通知只是一个片断,所以把这个片断存入数据库,等待后面的片断。可后面永远都不会再有任何相关的片断了。。。。。。( n  n; o9 z: K4 U* s, ~
  q! z5 K8 f* Y
知道了原因,无论解决方案有多困难,也总算是有解决的希望了。
; u7 [' c9 |/ h6 g- g1 h * i& H" l1 v1 s- j+ v8 [
解决方案用java写出来是这样的:
  1. userData = sms.getUserData();
    7 v2 X/ r) ]1 {6 U: Z+ C1 I
  2. BitwiseInputStream bis = BitwiseInputStream.new(userData);
    + X# h) p; C5 J) L4 r  }8 ?3 I
  3. bis.skip(69);1 J  g) \) r" x7 i+ H
  4. userData = bis.readByteArray(userData.length * 8 - 72);
    7 B% ~2 l1 W- ~  f# K9 |3 T
  5. processCdmaWapPdu(userData,......)
    ' L/ [! ^- i$ U! P! V

  6. 0 C: b/ t7 K$ U4 W
复制代码
意思就是说,把byte[]数组的userData转化成bit流,跳过最前面的69个bit(头部数据),取bit流(CHARi数据),扔掉最后的3个bit的000填充位,把取到的bit流再转化成byte[]数组。
& A9 I! W% U. g8 E! ?: T; O - v+ \( i( D2 N6 x2 @' J4 T
然后,把这段java代码人肉编译成dalvik虚拟机的字节码。。。。。。。

  1. 6 o0 F3 Y; J( T, j/ C
  2. move-object/from16 v11, v10  # v10: userData5 |1 q4 p% q3 Z1 W+ n$ k
  3. array-length v11, v11  # v11: userData.length
    * z% a( R- Y" ]" Y5 b$ g! F4 w

  4. 9 E: G, i. c+ {$ ?
  5. new-instance v12, Lcom/android/internal/util/BitwiseInputStream;
    0 j; A; {+ V& `. I% O9 H
  6. invoke-direct {v12, v10}, Lcom/android/internal/util/BitwiseInputStream;-><init>([B)V  # v12: bis- H  |' o% u# p! j9 C. c
  7. ; q# A2 M- |% F# Y4 L7 w3 U
  8. const/16 v8, 0x45+ E# @, M$ R; g* z" |6 M- K
  9. invoke-virtual {v12, v8}, Lcom/android/internal/util/BitwiseInputStream;->skip(I)V. h$ L' g( E% ?* V1 X" ?" H% {

  10. ! z; u0 ^5 p4 J7 I" @
  11. mul-int/lit8 v11, v11, 0x8$ \3 U& q' Y" w6 m. |, s# m
  12. add-int/lit8 v11, v11, -0x48  # userData.length*8 - 72
    4 \: {) I) H$ k% w* Y  J! B

  13. . `& u( \0 B7 D" q% Z$ p( s
  14. invoke-virtual {v12, v11}, Lcom/android/internal/util/BitwiseInputStream;->readByteArray(I)[B
    * g. X! R1 @" T! _3 \9 r# J/ y' {
  15. move-result-object v10( @! m! D, A1 G7 V
复制代码
编译,替换,重启,搞定。& J3 `2 O- d7 @9 U+ V' {
1 ^: k9 |2 {- Y4 N
事后,我有一点点小纳闷,这套原始代码是美版ROM用的,难道美帝国主义CDMA运营商的彩信,传过来的userData,直接就是CHARi吗,不包括头部的数据?否则按照ROM中的代码,他们也肯定是无法接收彩信的。这个问题看来还要美帝自己来回答了。我们自己的好用就行,不去解放水深火热受剥削受压迫中的美国人民了。。。。。。& o1 I9 ^9 v7 T- Q5 J
, U: a2 d  i( u

# Z+ ^3 F- P  R' p- j3 z: U& J' |( b3、自动接收和发送彩信' W3 [8 j  D$ Z% @. T) m1 ]

% ^$ ^5 u+ C: B, Y, m, U, Q搞定了彩信的接收,现在发送和接收都没有问题了。但还有一个小小的缺憾,就是发送和接收彩信时,需要先手动连接3G,不能做到自动收发。5 f* B$ C7 E( p& h! W

! b8 O+ ^2 I3 w4 r! z; t* o除了我个人的完美主义倾向在作怪之外,我承认我也喜欢上了这种修改工作,身心俱疲的同时,身心俱爽。。。。。。" k+ O2 ^  ]' [* F! p- ^- \

8 f" S- a- f+ x( c# Y3.1 不能自动收发的原因
2 O9 z8 z, E/ d$ a" s: ^  g9 B2 W ( {' E* M  L/ D5 a
经过一段时间的研究(这个研究时间倒不是很长,比上面研究彩信接收的时间短多了),发现在收发彩信时,系统会自动进行3G网络连接,然后另起一个线程收发彩信。. z/ t/ t6 C% \& `9 ?  x5 \! a

1 h/ P" j7 m/ |9 {但关键问题是,3G网络连接是需要时间的,在ppp拨号的客户端还在和服务端卿卿我我互传数据包确认关系的时候,收发彩信的线程就已经开始工作了,所以当然,在连接10.0.0.200的MMSC时,是会被refused的,然后程序抛出ioexception异常,收发事务被标记失败,最后系统断开3G连接。彩信的收发企图就这样被扼杀了。
3 g* z* K/ B3 X% c3 h + ?% W! |3 X5 \6 _; A
3.2 解决方案
0 \2 ?! @1 B* _  [5 W( Z5 k1 R! ~ 4 Y$ [( N0 s' c' F  h  i
知道了原因,再解决就只是时间问题了。解决方案有很多种,我采用的是,在getPdu和sendPdu这两个方法的最前部,加入ensureRouteToHost方法,同时修改ensureRouteToHost,加入一个带有超时退出的循环,判断requestRouteToHost的值。
) g0 v5 t8 E* J) q' i
- m" D6 }3 W( x! z0 g如果3G尚未连接,就sleep(1000)之后再试,直到从route上判断3G建立连接成功,退出ensureRouteToHost,进入收发彩信的流程。' [# v( ?& h7 T

  B, W5 _4 ]2 ]3 L* B0 V" ]如果60秒内3G仍未成功建立连接(要么信号不好,要不就是你欠费了。。。。。。),抛出ioexception异常,通知系统收发彩信失败,系统自动进入内置的重试排期计划。
! M. j; F$ W4 C: c5 ] " Q* N( Q4 I) V! D
于这种循环写起来比较麻烦,所以没有写java,直接写dalvik字节码:
  1. / j' A4 s* @# {. e1 v. D

  2. 3 w3 d! i& ~6 w3 _
  3. const/16 v10, 0x3c' U) R9 ?. x7 }* v5 X  F

  4. ) G* z; N* E2 t9 k! a, U
  5. :cond_46 ]. \$ t; G  O  _+ b, P9 n
  6. add-int/lit8 v10, v10, -0x1* g: e) P! p+ d* m
  7. if-eqz v10, :cond_5
    2 W, F) Z' S6 }: g/ B, Q& q8 y9 x
  8. 7 F4 U( _: m6 h: Y) k
  9. invoke-virtual {v0, v7, v1}, Landroid/net/ConnectivityManager;->requestRouteToHost(II)Z, g0 h8 _" U2 z1 |- f6 ]9 h( _

  10. - z9 V2 f* L, `/ g
  11. move-result v42 g; t- Y$ Y+ D

  12. 7 c4 R8 A% [/ o4 x: s; n
  13. if-nez v4, :cond_3
    * C5 m( E5 c7 |& v
  14. ) W# q3 t# T. o3 O
  15. const-wide v11, 0x3e87 t, f2 A, ^: ?/ [" L# E  V
  16. invoke-static {v11, v12}, Ljava/lang/Thread;->sleep(J)V
    0 ~4 _  r' ]/ u6 \/ P4 l, p4 C# v
  17. ; I8 J% Y6 l8 h: m$ Y" B
  18. goto :cond_48 V% a- v' x+ X) F8 E, L

  19. ! A7 z  ]# w+ i' h, W/ F
  20. :cond_5
    & Y, {% j7 f- J! t( Z
复制代码
目前在我本机测试的情况,可以完美收发彩信。在3G关闭的情况下,发送和接收彩信,系统可自动连接3G,收发成功后自动断开,极大地满足了我个人的完美主义倾向和老清新风格。。。。。。# A5 _+ {) w7 V% j) w7 F+ \
回复

使用道具 举报

发表于 2012-10-9 20:03 | 显示全部楼层
好文章,支持
回复 支持 反对

使用道具 举报

wengyq 该用户已被删除
发表于 2014-5-9 10:19 | 显示全部楼层
提示: 作者被禁止或删除 内容自动屏蔽
回复 支持 反对

使用道具 举报

发表于 2015-5-14 08:30 | 显示全部楼层
牛人一个,支持!
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 加入我们

本版积分规则

QQ|Archiver|手机版|小黑屋|吹友吧 ( 京ICP备05078561号 )

GMT+8, 2024-11-19 19:17 , Processed in 0.300082 second(s), 17 queries .

Powered by Discuz! X3.5 Licensed

© 2001-2024 Discuz! Team.

快速回复 返回顶部 返回列表