美文网首页
Android非主用户无线相关设置的显示

Android非主用户无线相关设置的显示

作者: wbxjack | 来源:发表于2017-05-12 14:52 被阅读221次

    需求

    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非主用户无线相关设置的显示

          本文链接:https://www.haomeiwen.com/subject/fzogtxtx.html