Android非主用户无线相关设置的显示
需求
Android多用户下,要让每个用户都能看到移动网络相关设置(首先网络类型/APN等)。
现状
- Android默认的多用户实现中,只有主用户(primary user)是可以看到的,如果新建一个另外的用户,则没有权限去看这些东西,参见WirelessSettings.java中的判断(boolean isSecondaryUser = myUserId != UserHandle.USER_OWNER;如果是SecondaryUser,则不显示相关设置)。
- Primary user即user 0,是指开机进入的那个用户,这个用户的权限是最高的,framework,phone和其他的所有core service都是在这个用户下创建和初始化的。
- 无线相关(即phone对象和无线相关的服务),PhoneApp初始化的时候,会判断是否在primary user(见下面),如果是user 0,那么会初始化这些对象;如果非0用户,则会在此用户创建一个新的空实现的Phone对象,因为在oncreate里面什么都没做。所以不能在另外的用户中直接调用phone相关的对象。
if (UserHandle.myUserId() == 0) {
// We are running as the primary user, so should bring up the
// global phone state.
mPhoneGlobals = new PhoneGlobals(this);
mPhoneGlobals.onCreate();
mTelephonyGlobals = new TelephonyGlobals(this);
mTelephonyGlobals.onCreate();
}
方案
针对“现状”中描述的情况,我们要在非primary user中显示无线相关的信息,我们需要一个途径来获得我们需要的信息。有两个解决办法:
- 放开PhoneApp对于用户的限制,在非primary user中也初始化好phone相关对象。
- 通过扩展primary user中的phone service来让其他用户获得相关信息。
评估
方案1:在非primary user中也初始化phone相关对象。由于phone对象作为核心对象,从中初始化了很多其他相关的对象(例如sim卡,RIL等),这样就涉及从上到下要全部打通才行,难度颇大。尤其是Android5.1-7.1中phone重构颇多,这个作为终极解决方案,但是在时间紧迫的前提下,我们只是简单尝试,没事深入,感觉坑很多。后续会继续探索,争取在最新的安卓版本上搞定这个大事哈。感兴趣的可以参考下面的链接,作者是在android4.2上做的,基本思路也是一致的。剩下的工作就是安卓版本差异上带来的难度。
《android源码探索----多用户下phone进程问题》
http://blog.csdn.net/stephen8341/article/details/38079679
方案2:扩展“phone” service,让其他user可以访问到相关信息即可,毕竟只是setting展示,基本上需要的就是phone id/sub id之类的。这是个取巧的办法,不完善,是在时间和项目的压力下,妥协的方案。
方案2的实现
user 0中已经在开机的时候启动的phone service(packages/services/Telephony/src/com/android/phone/PhoneInterfaceManager.java 的publish()函数),
private void publish() {
if (DBG) log("publish: " + this);
ServiceManager.addService("phone", this);
}
在其他user中需要使用的时候,通过如下方法调用
ITelephony phone = ITelephony.Stub.asInterface(ServiceManager.checkService("phone"));
我们的做法就是扩展这个phone service,需要改动如下几个文件:
- frameworks/base/telephony/java/com/android/internal/telephony/ITelephony.aidl 定义service提供的函数的aidl,把扩展的函数加进去
- packages/services/Telephony/src/com/android/phone/PhoneInterfaceManager.java 这个ITelephoy.aidl的实现
- 对于要回传结果的场景,例如设置网络,新建ITelephonyAdapter.aidl文件,定义回掉接口。并把这个aidl加入到framework/base下的Android.mk文件中,保证系统会编译这个文件。
- 添加ITelephonyAdapter.aidl的实现文件TelephonyAdapter.java
- 剩下的就是判断当前user,如果是非主用户,通过service取相关信息,然后显示相关UI。
总结
这里只是记录一下解决问题的思路,思路对了,再慢也会完成相关功能。如果一开始就选了一个走不通的路,那么再努力也是徒劳。