telephony

相关代码

frameworks/base/telephony --- telephony framework
frameworks/base/telecomm --- telecomm framework 没有详细看
package/services/telephony --- teleService.apk
package/services/Telecomm --- telepcomm.apk 没有详细看
frameworks/opt/telephony --- telephony-common

Telephony流程.png

TELEPHONY_SERVICE

系统启动后,由systemserver.java的main方法开始:

systemserver.java
main() -> run() -> startBootstrapServices(其中启动AMS)
->startotherServices -> AMS. systemReady 中调用startPersistentApps去启动phone应用:

phoneapp-> 
phoneglobals->onCreat() -> 
PhoneInterfaceManager.init(this) -> public() ->
TelephonyFrameworkInitializer
                .getTelephonyServiceManager()
                .getTelephonyServiceRegisterer()
                .register(this);
//看下具体的实现
public ServiceRegisterer getTelephonyServiceRegisterer() {
        return new ServiceRegisterer(Context.TELEPHONY_SERVICE);
 }

如上,在启动phone过程中通过register操作,将PhoneInterfaceManager注册成为真正的Context.TELEPHONY_SERVICE的service端。
一个细节问题,上述的getTelephonyServiceManager是在什么时候赋值的?通过搜索setTelephonyServiceManager方法,发现是在ActivityThread中的main方法中:

调用initializeMainlineModules() ->
TelephonyFrameworkInitializer.setTelephonyServiceManager(new TelephonyServiceManager());

TelephonyServiceManager是用于管理和telephony相关的各种service,主要是确保telephony相关的各种service能够注册成功。

SystemServiceRegistry

SystemServiceRegistry用于在APP端缓存各种系统服务,方便调用系统服务。其static方法如下:

       static {   
            //......
           JobSchedulerFrameworkInitializer.registerServiceWrappers();
           BlobStoreManagerFrameworkInitializer.initialize();
           TelephonyFrameworkInitializer.registerServiceWrappers();
           WifiFrameworkInitializer.registerServiceWrappers();
           StatsFrameworkInitializer.registerServiceWrappers();
           //......
       } 


   // TelephonyFrameworkInitializer 的 registerServiceWrappers方法具体实现:
   /**
    * Called by {@link SystemServiceRegistry}'s static initializer and registers all telephony
    * services to {@link Context}, so that {@link Context#getSystemService} can return them.
    *
    * @throws IllegalStateException if this is called from anywhere besides
    * {@link SystemServiceRegistry}
    */
   public static void registerServiceWrappers() {
       SystemServiceRegistry.registerContextAwareService(
               Context.TELEPHONY_SERVICE,
               TelephonyManager.class,
               context -> new TelephonyManager(context)
       );
       SystemServiceRegistry.registerContextAwareService(
               Context.TELEPHONY_SUBSCRIPTION_SERVICE,
               SubscriptionManager.class,
               context -> new SubscriptionManager(context)
       );
       SystemServiceRegistry.registerContextAwareService(
               Context.CARRIER_CONFIG_SERVICE,
               CarrierConfigManager.class,
               context -> new CarrierConfigManager(context)
       );
       SystemServiceRegistry.registerContextAwareService(
               Context.EUICC_SERVICE,
               EuiccManager.class,
               context -> new EuiccManager(context)
       );
       SystemServiceRegistry.registerContextAwareService(
               Context.EUICC_CARD_SERVICE,
               EuiccCardManager.class,
               context -> new EuiccCardManager(context)
       );
       SystemServiceRegistry.registerContextAwareService(
               Context.TELEPHONY_IMS_SERVICE,
               ImsManager.class,
               context -> new ImsManager(context)
       );
   }

那么SystemServiceRegistry的调用是在什么时候呢?

    //ConextImpl.java
    final Object[] mServiceCache = SystemServiceRegistry.createServiceCache();

当ConextImpl对象创建时,其属性就会创建,由于上述所说,SystemServiceRegistry有static代码块,因此该代码块会在第一次使用时就会加载。即在ConextImpl对象创建好之前,相关的服务就已经在其中注册了。

Telcommservice

systemserver.java
main() -> run() -> startBootstrapServices(其中启动AMS)
->startotherServices -> 去StartTelecomLoaderService

            t.traceBegin("StartTelecomLoaderService");
            mSystemServiceManager.startService(TelecomLoaderService.class);
            t.traceEnd();

后面当完成开机后,在AMS的finishBooting中去执行
// Let system services know.
mSystemServiceManager.startBootPhase(t, SystemService.PHASE_BOOT_COMPLETED);
这个方法中回去回调service.onBootPhase(mCurrentPhase)。
继续查看TelecomLoaderService的onBootPhase实现:

    @Override
    public void onBootPhase(int phase) {
        if (phase == PHASE_ACTIVITY_MANAGER_READY) {
            registerDefaultAppNotifier();
            registerCarrierConfigChangedReceiver();
            // core services will have already been loaded.
            setupServiceRepository();
            connectToTelecom();
        }
    }
    private void connectToTelecom() {
        synchronized (mLock) {
            if (mServiceConnection != null) {
                // TODO: Is unbinding worth doing or wait for system to rebind?
                mContext.unbindService(mServiceConnection);
                mServiceConnection = null;
            }

            TelecomServiceConnection serviceConnection = new TelecomServiceConnection();
            Intent intent = new Intent(SERVICE_ACTION);
            intent.setComponent(SERVICE_COMPONENT);
            int flags = Context.BIND_IMPORTANT | Context.BIND_FOREGROUND_SERVICE
                    | Context.BIND_AUTO_CREATE;

            // Bind to Telecom and register the service
            if (mContext.bindServiceAsUser(intent, serviceConnection, flags, UserHandle.SYSTEM)) {
                mServiceConnection = serviceConnection;
            }
        }
    }
    private static final ComponentName SERVICE_COMPONENT = new ComponentName(
            "com.android.server.telecom",
            "com.android.server.telecom.components.TelecomService");

    private static final String SERVICE_ACTION = "com.android.ITelecomService";

可以看到TelecomLoaderService去bind了TelecomService,TelecomService做为TelecomLoaderService的服务端,实现了ITelecomLoader.Stub:

public class TelecomService extends Service implements TelecomSystem.Component {

    @Override
    public IBinder onBind(Intent intent) {
        Log.d(this, "onBind");
        //Task9999555-zhengpinhuang@tcl.com-Mobile data off-begin
        TelecomGlobals.createInstance(getApplicationContext());
        CarrierConfigManagerEx.init(getApplicationContext());
        //Task9999555-zhengpinhuang@tcl.com-Mobile data off-end
        return new ITelecomLoader.Stub() {
            @Override
            public ITelecomService createTelecomService(IInternalServiceRetriever retriever) {
                InternalServiceRetrieverAdapter adapter =
                        new InternalServiceRetrieverAdapter(retriever);
                initializeTelecomSystem(TelecomService.this, adapter);
                synchronized (getTelecomSystem().getLock()) {
                    return getTelecomSystem().getTelecomServiceImpl().getBinder();
                }
            }
        };
    }
}

接下去看下,TelecomLoaderService在ServiceConnected成功后做了什么?

    private class TelecomServiceConnection implements ServiceConnection {
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            // Normally, we would listen for death here, but since telecom runs in the same process
            // as this loader (process="system") that's redundant here.
            try {
                ITelecomLoader telecomLoader = ITelecomLoader.Stub.asInterface(service);
                ITelecomService telecomService = telecomLoader.createTelecomService(mServiceRepo);

                SmsApplication.getDefaultMmsApplication(mContext, false);
                ServiceManager.addService(Context.TELECOM_SERVICE, telecomService.asBinder());

                //......省略部分实现
               
            } catch (RemoteException e) {
                Slog.w(TAG, "Failed linking to death.");
            }
        }

可以看到在拿到service后,第一时间就是调用createTelecomService,真正的实现是在TelecomLoaderService的service端,就是上面提到的实现了ITelecomLoader.Stub的地方。

        return new ITelecomLoader.Stub() {
            @Override
            public ITelecomService createTelecomService(IInternalServiceRetriever retriever) {
                InternalServiceRetrieverAdapter adapter =
                        new InternalServiceRetrieverAdapter(retriever);
                initializeTelecomSystem(TelecomService.this, adapter);
                synchronized (getTelecomSystem().getLock()) {
                    return getTelecomSystem().getTelecomServiceImpl().getBinder();
                }
            }
        };

getTelecomServiceImpl的实现可以看到:

    public TelecomServiceImpl getTelecomServiceImpl() {
        return mTelecomServiceImpl;
    }

mTelecomServiceImpl是在TelecomSystem的构造方法里面new的。再看下getBinder的实现:

    public ITelecomService.Stub getBinder() {
        return mBinderImpl;
    }
    private final ITelecomService.Stub mBinderImpl = new ITelecomService.Stub() {
        @Override
        public PhoneAccountHandle getDefaultOutgoingPhoneAccount(String uriScheme,
                String callingPackage, String callingFeatureId) {
}

可以看到这里是真正telecomservice的实现端,是一个匿名内部类。再一步步往回走,在拿到真正的telecomservice后,通过ServiceManager.addService将其加入到ServiceManager中。

方法总结

上面总结了很多系统service的启动和注册流程,那么对于这些系统的service,如何在看源码的时候快速定位到具体service的实现呢?在已经知道答案的情况下,我个人大致反推了一下,方便后续查看其它系统service时能够快速定位。
1、假设现在要Context.TELECOM_SERVICE的具体实现,从代码可以知道这个service对应的标识是:

    public static final String TELECOM_SERVICE = "telecom";

2、adb shell service list|grep -rin telecom 结果如下:

telecom: [com.android.internal.telecom.ITelecomService]
看到对应的aidl接口是ITelecomService.aidl

3、代码全局搜索"extends ITelecomService.stub" ,如果没有再搜索 "new ITelecomService.stub" ,因为有部分service端是匿名的类,telcom service就是这样的。

完成上面3步,这些系统service的真正实现的地方基本就定位到了。

参考内容

如何实现服务器端调用应用端的例子。
https://cloud.tencent.com/developer/article/1600478
解析telephonymanager和telecom的文章,里面的很多时序图很好。
https://blog.csdn.net/jason_wzn/article/details/58164251
https://blog.csdn.net/ZhongGuoRenMei/article/details/97037870

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,047评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,807评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,501评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,839评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,951评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,117评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,188评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,929评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,372评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,679评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,837评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,536评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,168评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,886评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,129评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,665评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,739评论 2 351

推荐阅读更多精彩内容