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,需要改动如下几个文件:

  1. frameworks/base/telephony/java/com/android/internal/telephony/ITelephony.aidl 定义service提供的函数的aidl,把扩展的函数加进去
  2. packages/services/Telephony/src/com/android/phone/PhoneInterfaceManager.java 这个ITelephoy.aidl的实现
  3. 对于要回传结果的场景,例如设置网络,新建ITelephonyAdapter.aidl文件,定义回掉接口。并把这个aidl加入到framework/base下的Android.mk文件中,保证系统会编译这个文件。
  4. 添加ITelephonyAdapter.aidl的实现文件TelephonyAdapter.java
  5. 剩下的就是判断当前user,如果是非主用户,通过service取相关信息,然后显示相关UI。

总结

这里只是记录一下解决问题的思路,思路对了,再慢也会完成相关功能。如果一开始就选了一个走不通的路,那么再努力也是徒劳。

你可能感兴趣的:(Android非主用户无线相关设置的显示)