Android 5.0 双卡信息管理分析

首先,如前面的博文所讲的,Android5.0开始支持双卡了。另外,对于双卡的卡信息的管理,也有了实现,尽管还不是完全彻底完整,如卡的slot id, display name,iccid,color等,其设计思路竟然跟之前接触到的一个平台是一样的,都是同不同颜色来标识不同的卡,让用户一目了然,只是5.0的实现目前还局限在FW框架里,应用层的实现还没有,相信,等到5.1或者再之后的版本中,我们就可以在setting里看到对卡表示颜色、名称等进行设置的功能菜单啦。

下面进入正题,来分析Android5.0的双卡信息是如何来维护的:

1 卡信息的存储

先说结论,5.0是将所有卡信息通过数据库进行保存的。数据库表URL:content://telephony/siminfo,对应到的数据库为telephony.db中的siminfo表,代码文件在TelephonyProvider.java,在这里可以看到createTable的详情,来看看这个表都包含哪些字段。

总共设计了11个字段,下面这张图是从5.0模拟器上拉出来的telephony.db里截取到的,可以用来做示例:

表名:

表数据示例:

下面分别介绍一下这些字段的含义。

1.1 _id

这个自不必说,android里的每个db都有的,是数据库里的数据主key,这里的_id值也是以后代码里看到的subId,是在数据库中保存时所在的条目的唯一标识。有多少条数据,就说明保存了多少个sim卡的信息;

1.2 icc_id

从卡上读取得到,是卡的唯一身份标识,世界上的所有sim卡,每个卡都有不同的iccid,就像身份证一样;如上面图中的16进制串89014103211118510720;

1.3 sim_id

分配给卡的id序号,android设计从0开始,最大为卡槽个数,比如如果是双卡终端,那么只有可能是0或者1;当然无卡时其值为-1,所以这里其实是与卡槽固定对应的,卡槽如果有卡就取对应id,如果无卡,就设为-1;

1.4 display_name

分配给卡的显示名字,从代码上来看开机后会尝试使用运营商名字,如果取不到,就使用简单的SUB 01这样的字串表示,等拿到运营商名字之后重新set,另外从下面的name_source字段的设计来看,android是允许用户来自己指定这个显示名的;

1.5 name_source

表明display_name字段的来源,有两种来源,一是系统自动,name_source取值为0,另一种就是来自用户指定,取值为1;


/**

* The name_source is the default

* @hide

*/

publicstaticfinalint NAME_SOURCE_DEFAULT_SOURCE =0;

/**

* The name_source is from the SIM

* @hide

*/

publicstaticfinalint NAME_SOURCE_SIM_SOURCE =1;

/**

* The name_source is from the user

* @hide

*/

publicstaticfinalint NAME_SOURCE_USER_INPUT =2;

1.6 color

显示颜色,将使用颜色在UI上明显区分卡1和卡2;android设定每个卡都只能从固定的几个颜色中取,如下:


/** @hide */

publicstaticfinal String COLOR ="color";

/** @hide */

publicstaticfinalint COLOR_1 =0;

/** @hide */

publicstaticfinalint COLOR_2 =1;

/** @hide */

publicstaticfinalint COLOR_3 =2;

/** @hide */

publicstaticfinalint COLOR_4 =3;

/** @hide */

publicstaticfinalint COLOR_DEFAULT = COLOR_1;

这四种颜色根据主题是dark还是light分别对应4种颜色,这些颜色其实并非是色值,而是固定颜色的背景9.png图片,都定义在资源drawable里,看来是用来做backgroud用的。如

sim_dark_blue sim_light_purple

其他几种颜色可以参考 SubscriptionController. setSimResource函数里去color资源数组的初始化。

1.7 number

该卡对应的号码,phone number。 //TODO:研究一下这个是怎么获取到呢?

1.8 display_number_format

标识number字段的格式,总共有固定的3种,取其一。


/** @hide */

publicstaticfinalint DISPLAY_NUMBER_NONE =0;

/** @hide */

publicstaticfinalint DISPLAY_NUMBER_FIRST =1;

/** @hide */

publicstaticfinalint DISPLAY_NUMBER_LAST =2;

/** @hide */

publicstaticfinalint DISLPAY_NUMBER_DEFAULT = DISPLAY_NUMBER_FIRST;

从SubscriptionController. setDisplayNumberFormat()函数的注释可以看出点门道,

*Set number display format. 0: none, 1: the first four digits, 2: the last four digits

但搜了一下,对应的setDisplayNumber()函数并没有跟这个format有关联,直接保存传入的number到数据库了。所以还不太清楚the first four digits 和 the last four digits的具体含义,等以后android应用层完善后应该能看到具体的使用方式。

1.9 data_roaming

是否允许这张卡进行数据漫游,默认禁止漫游。


/** @hide */

publicstaticfinalint DATA_ROAMING_ENABLE =1;

/** @hide */

publicstaticfinalint DATA_ROAMING_DISABLE =0;

/** @hide */

publicstaticfinalint DATA_ROAMING_DEFAULT = DATA_ROAMING_DISABLE;

1.10 mcc

移动国家码,从卡上读取得到,卡的mcc码;

1.11 mnc

移动网络码,从卡上读取得到,卡的mnc码。度娘到对于mcc和mnc的解释:


MCC 是 Mobile Country Code 的缩写,译为移动国家代码。它由三位数字组成。用于标识一个国家,但一个国家可以被分配多个 MCC 。比如美国的 MCC 有 310,311,和316。中国的 MCC 只有 460。

MNC 是 Mobile Network Code 的缩写,译为移动网络代码。它由二到三位数字组成。MNC和 MCC 合在一起唯一标识一个移动网络提供者。比如中国移动的 MNC 是 00,中国联通的 MNC 是 01,中国联通 CDMA 的 MNC 是 03,中国卫星全球星网的 MNC 是 04。

因此,460 00 就唯一标识了中国移动。

所以卡的displayname应该也是通过mcc+mnc得到的。

2 主要工作类

上面介绍了卡信息的数据设计,接下来看看有哪些类围绕着这些数据来对外提供接口和功能。



功能介绍


TelephonyProvider


siminfo数据库provider,直接操作DB,实现siminfo表的增删改查。


SubscriptionController


实现为远程service,在phone初始化时被创建,作为数据库的对外接口提供功能,其内部维护从数据库读取到的siminfo list,并实现了大量的如getSubId getDisplayName setDisplayName等getter和setter方法,这些方法维护siminfo list以及通过URL访问数据库来实现功能。


SubscriptionManager


该类所有的内部接口都实现为了static,也就是该类是一个纯粹提供接口的静态类,其内部并不实现具体的逻辑,只是通过调用SubscriptionController这个service的对应接口来完成工作,所以该类是一个对外接口的封装,APP可以直接通过SubscriptionManager.getSubId等来实现功能,不需要直接操作service。目前还是Hide的,估计以后会开放给SDK。这种设计模式,android FW层用的很多,如SmsMnager、TelephonyManager等。


SubInfoRecordUpdater


卡信息变化的监听者和更新者,其在PhoneFactory中创建完phones之后被创建,起来之后会注册对 ACTION_SIM_STATE_CHANGED 的监听,收到监听之后更新对应siminfo的信息,如iccid、displayname(运营商名)、phonenumber。并且在自己内部维护了每个卡的卡状态,这里的卡状态区别与ACTION_SIM_STATE_CHANGED所携带的卡状态,ACTION_SIM_STATE_CHANGED的卡状态有 LOCKED、READY、NOT READY、ABSENT等,是指的卡的具体状态,而这里的卡状态是指 有没有插卡、该卡槽的卡是不是变过了(换了一张卡)、卡槽的卡是一张没见过的新卡、卡槽的卡是位置互换啦、卡槽的卡没变化。【状态还真有点绕,管的挺多的啊。。。】这些状态是在收到ACTION_SIM_STATE_CHANGED之后,根据卡状态以及卡的iccid等一些信息算出来的,如果发现有新卡插入会进行广播。


SubInfoRecord


subinfo的可序列化实现,封装数据库数据。

另外,在代码中还可以看到 Subscription 和 SubscriptionData这两个类,这两个其实只是数据封装,并不直接跟卡信息管理相关,其只是保存了一些卡的数据来提供给phone等使用,目前还找不到组织他们的接口,全局搜索会发现,好多地方google都注释掉了,逻辑还没完整。SubscriptionData是Subscription list,目前从代码来看,这两个还没有真正用起来,所以可以先无视了,不过要吐槽下google的代码,起名咋这么绕啊。另外还有个 CdmaSubscriptionSourceManager的类,这个是跟CDMA的卡提供者信息有关的,会影响到具体的网络行为,也跟本文的双卡信息管理无关。

3 工作流程

通过上面主要类的介绍,基本上整个双卡信息管理的框架就出来了,画了一个图方便理解。

4总结

5.0通过DB来保存和维护双卡信息,SubInfoRecordUpdater通过监听卡的变化动态更新数据库里的卡数据;SubscriptionController和SubscriptionManager通过各种接口来向外提供卡信息的查询和修改,这样看来逻辑还是比较清晰明了的。

时间: 2024-08-02 06:50:48

Android 5.0 双卡信息管理分析的相关文章

Android 5.0 Uicc框架分析

Uicc框架 UICC框架是Android在4.1引入的,使的对卡的管理控制更加清晰.要了解这个UICC框架,需要从UiccController开始,它是整个UICC框架的开始与控制者,该类被设计为单例,是消息处理类Handler的子类,所以其实现肯定是基于event触发的,其在Phone创建的早期被初始化: 1 2 3 // Instantiate UiccController so that all other classes can just // call getInstance() m

Android 6.0 Overview Screen实现原理

Android 4.0中添加了一个很有用的特性,那就是overView Screen功能,也就是最近任务预览功能.这个功能提供了一个列表试图,方便用户简单快捷地了解到最近使用的app或者最近进行的任务.这个功能和iOS的最近任务在界面上很相似.在android 5.0中,这个任务得到了进一步的加强,在android 5.0之前overView Screen中显示的任务快照是不可以配置的,但是在android 5.0中是可以配置的,开发者可以指定那些activity以什么样的形式,什么UI风格显示

Android 5.0 怎样正确启用isLoggable(二)__原理分析

前置文章 <Android 5.0 怎样正确启用isLoggable(一)__使用具体解释> 概要 在上文<Android 5.0 怎样正确启用isLoggable(一)__使用具体解释>中分析了isLoggable的用法,本文主要分析isLoggable实现原理以及user版系统root后永久enable isLoggable的原理,并使用脚本自己主动设置isLoggable相关属性. 本文来自http://blog.csdn.net/yihongyuelan 转载请务必注明出处

Android 4.0 Launcher源码分析系列(一)

从今天起傻蛋打算做一个系列文章,对最新的Android 4.0 系统中的Launcher,也就是Android 4.0原生的桌面程序,进行一个深入浅出的分析,从而引领Android系统的编程爱好者对 Launcher的设计思想,实现方式来做一个研究,从而能够通过这个实例最掌握到目前世界领先的设计方法,同时在程序中加入我们的一些新的实现.众所周知,对一些优秀源代码的分析,是提高编程水平的一条便捷的方式,希望本系列文章能够给大家带来一定的启发,同时欢迎大家和作者一起讨论,作者的微博是:http://

Android 5.0 如何正确启用isLoggable(二)__原理分析

前置文章 <Android 5.0 如何正确启用isLoggable(一)__使用详解> 概要 在上文<Android 5.0 如何正确启用isLoggable(一)__使用详解>中分析了isLoggable的使用方法,本文主要分析isLoggable实现原理以及user版系统root后永久enable isLoggable的原理,并使用脚本自动设置isLoggable相关属性. 本文来自http://blog.csdn.net/yihongyuelan 转载请务必注明出处 isL

android 6.0权限全面详细分析和解决方案

原文: http://www.2cto.com/kf/201512/455888.html http://blog.csdn.net/yangqingqo/article/details/48371123 http://inthecheesefactory.com/blog/things-you-need-to-know-about-Android-m-permission-developer-edition/en 一.Marshmallow版本权限简介 android的权限系统一直是首要的安全

Android 4.0 ICS SystemUI浅析——StatusBar结构分析

Android 4.0 ICS SystemUI浅析——StatusBar结构分析 分类: Android2012-06-30 14:45 23687人阅读 评论(8) 收藏 举报 androidsignal代码分析iconseclipse平台 在上一篇文章<Android 4.0 ICS SystemUI浅析——SystemUI启动流程>中以及提到了SystemUI的组成,本文主要分析其中的StatusBar结构. 1.布局概览 首先,我们通过hierarchyviewer这个工具来查看一下

【腾讯优测干货分享】Android5.0-6.0双卡适配指南

作者:腾讯优测开发工程师 于长敏 这里仅以获取sim卡的IMSI接口(getSubscriberId)和发短信接口(sendTextMessage)为例来详细讲解一下Android5.0-6.0双卡适配的策略,其他方面的双卡适配方案跟4.4以前相比并无特别大的区别,之前我们已有专家对此进行过详细的总结,这里就不重复说明了. 从Android5.0开始,加入了对双卡的管理: 首先从数据库方面来看,其设计思路跟以前某平台是一样的,加入一个siminfo数据表到telephony.db来管理双卡的信息

Android 7.0 Nougat(牛轧糖)---对开发者来说

android 7.0出来了.让你的app准备迎接最新的android版本吧,支持节省电量和内存,这样新的系统行为.使用多窗口UI.直接恢复通知以及其他操作来扩展你的app. android 7.0介绍了各种各样的新功能给用户和开发者, 本文重点介绍面向开发者的一些新功能. 确保检查android 7.0的行为变化,了解有关平台的变化可能会影响你的app. 如果要了解更多的关于用户的新功能,请查看www.android.com. 1.支持多窗口 在android 7.0中,我们介绍了在支持多窗口