日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術(shù)文章
文章詳情頁

Android 系統(tǒng)服務(wù)TelecomService啟動過程原理分析

瀏覽:19日期:2022-09-23 11:03:55

由于一直負(fù)責(zé)的是Android Telephony部分的開發(fā)工作,對于通信過程的上層部分Telecom服務(wù)以及UI都沒有認(rèn)真研究過。最近恰好碰到一個通話方面的問題,涉及到了Telecom部分,因而就花時間仔細(xì)研究了下相關(guān)的代碼。這里做一個簡單的總結(jié)。這篇文章,主要以下兩個部分的內(nèi)容:

什么是Telecom服務(wù)?其作用是什么? Telecom模塊的啟動與初始化過程;

接下來一篇文章,主要以實際通話過程為例,分析下telephony收到來電后如何將電話信息發(fā)送到Telecom模塊以及Telecom是如何處理來電。

什么是Telecom服務(wù)

Telecom是Android的一個系統(tǒng)服務(wù),其主要作用是管理Android系統(tǒng)當(dāng)前的通話,如來電顯示,接聽電話,掛斷電話等功能,在Telephony模塊與上層UI之間起到了一個橋梁的作用。比如,Telephony有接收到新的來電時,首先會告知Telecom,然后由Telecom服務(wù)通知上層應(yīng)用來電信息,并顯示來電界面。

Telecom服務(wù)對外提供了一個接口類TelecomManager,通過其提供的接口,客戶端可以查詢通話狀態(tài),發(fā)送通話請求以及添加通話鏈接等。

從Telecom進(jìn)程對應(yīng)的AndroidManifest.xml文件來看,Telecom進(jìn)程的用戶ID跟系統(tǒng)進(jìn)程用戶ID相同,是系統(tǒng)的核心服務(wù)。那么,其中android:process='system'這個屬性值表示什么意思了?查看官方文檔,這個表示Telecom將啟動在進(jìn)程system中,這樣可以跟其他進(jìn)程進(jìn)行資源共享了(對于Android這個全局進(jìn)程,就是SystemServer所在的進(jìn)程)。

android:process

By setting this attribute to a process name that’s shared with another application, you can arrange for components of both applications to run in the same process — but only if the two applications also share a user ID and be signed with the same certificate.

If the name assigned to this attribute begins with a colon (‘:’), a new process, private to the application, is created when it’s needed. If the process name begins with a lowercase character, a global process of that name is created. A global process can be shared with other applications, reducing resource usage.

<manifest xmlns:android='http://schemas.android.com/apk/res/android' xmlns:androidprv='http://schemas.android.com/apk/prv/res/android' package='com.android.server.telecom' android:versionCode='1' android:versionName='1.0.0' coreApp='true' android:sharedUserId='android.uid.system'> <application android:label='@string/telecommAppLabel' android:icon='@mipmap/ic_launcher_phone' android:allowBackup='false' android:supportsRtl='true' android:process='system' android:usesCleartextTraffic='false' android:defaultToDeviceProtectedStorage='true' android:directBootAware='true'> .... // 包含TelecomService <service android:name='.components.TelecomService' android:singleUser='true' android:process='system'> <intent-filter> <action android:name='android.telecom.ITelecomService' /> </intent-filter> </service> .... </application> </manifest>

代碼路徑:

/android/applications/sources/services/Telecomm//android/frameworks/base/telecomm/

了解了什么是Telecom服務(wù)之后,就來看一看Telecom服務(wù)是如何啟動與初始化的。

Telecom進(jìn)程的啟動與初始化

在SystemServer進(jìn)程初始化完成啟動完系統(tǒng)的核心服務(wù)如ActivityManagerService后,就會加載系統(tǒng)其它服務(wù),這其中就包含了一個與Telecom服務(wù)啟動相關(guān)的系統(tǒng)服務(wù)專門用于加載Telecom:

private void startOtherServices() { .... //啟動TelecomLoaderService系統(tǒng)服務(wù),用于加載Telecom mSystemServiceManager.startService(TelecomLoaderService.class); // 啟動telephony注冊服務(wù),用于注冊監(jiān)聽telephony狀態(tài)的接口 telephonyRegistry = new TelephonyRegistry(context); ServiceManager.addService('telephony.registry', telephonyRegistry); }

調(diào)用系統(tǒng)服務(wù)管家SystemServiceManager的接口startService創(chuàng)建新的服務(wù),并注冊到系統(tǒng)中,最后調(diào)用onStart()啟動服務(wù)。

public class SystemServiceManager { @SuppressWarnings('unchecked') public SystemService startService(String className) { final Class<SystemService> serviceClass; try { serviceClass = (Class<SystemService>)Class.forName(className); } catch (ClassNotFoundException ex) { .... } return startService(serviceClass); } // 服務(wù)的class文件來創(chuàng)建新的服務(wù)對象(服務(wù)必須繼承SystemService) @SuppressWarnings('unchecked') public <T extends SystemService> T startService(Class<T> serviceClass) { try { final String name = serviceClass.getName(); Slog.i(TAG, 'Starting ' + name); Trace.traceBegin(Trace.TRACE_TAG_SYSTEM_SERVER, 'StartService ' + name); // Create the service. if (!SystemService.class.isAssignableFrom(serviceClass)) { throw new RuntimeException('Failed to create ' + name + ': service must extend ' + SystemService.class.getName()); } final T service; try { Constructor<T> constructor = serviceClass.getConstructor(Context.class); service = constructor.newInstance(mContext); } catch (InstantiationException ex) { throw new RuntimeException('Failed to create service ' + name + ': service could not be instantiated', ex); } .... // Register it. mServices.add(service); // Start it. try { service.onStart(); } catch (RuntimeException ex) { throw new RuntimeException('Failed to start service ' + name + ': onStart threw an exception', ex); } return service; } finally { Trace.traceEnd(Trace.TRACE_TAG_SYSTEM_SERVER); } } }

創(chuàng)建TelecomLoaderService系統(tǒng)服務(wù),將系統(tǒng)默認(rèn)的SMS應(yīng)用,撥號應(yīng)用以及SIM通話管理應(yīng)用(不知道這個什么鬼)告知PackageManagerService(PMS),以便在適當(dāng)?shù)臅r候可以找到應(yīng)用。

public class TelecomLoaderService extends SystemService { ... public TelecomLoaderService(Context context) { super(context); mContext = context; registerDefaultAppProviders(); } @Override public void onStart() { } private void registerDefaultAppProviders() { final PackageManagerInternal packageManagerInternal = LocalServices.getService( PackageManagerInternal.class); // Set a callback for the package manager to query the default sms app. packageManagerInternal.setSmsAppPackagesProvider( new PackageManagerInternal.PackagesProvider() { @Override public String[] getPackages(int userId) { synchronized (mLock) { .... ComponentName smsComponent = SmsApplication.getDefaultSmsApplication( mContext, true); if (smsComponent != null) { return new String[]{smsComponent.getPackageName()}; } return null; } }); // Set a callback for the package manager to query the default dialer app. packageManagerInternal.setDialerAppPackagesProvider( new PackageManagerInternal.PackagesProvider() { @Override public String[] getPackages(int userId) { synchronized (mLock) { .... String packageName = DefaultDialerManager.getDefaultDialerApplication(mContext); if (packageName != null) { return new String[]{packageName}; } return null; } }); // Set a callback for the package manager to query the default sim call manager. packageManagerInternal.setSimCallManagerPackagesProvider( new PackageManagerInternal.PackagesProvider() { @Override public String[] getPackages(int userId) { synchronized (mLock) { .... TelecomManager telecomManager = (TelecomManager) mContext.getSystemService(Context.TELECOM_SERVICE); PhoneAccountHandle phoneAccount = telecomManager.getSimCallManager(userId); if (phoneAccount != null) { return new String[]{phoneAccount.getComponentName().getPackageName()}; } return null; } }); } }

到目前,好像Telecom服務(wù)并沒啟動,那么究竟Telecom服務(wù)在哪里啟動的了?仔細(xì)看TelecomLoaderService的源代碼,其中有一個onBootPhase的函數(shù),用于SystemServer告知系統(tǒng)服務(wù)目前系統(tǒng)啟動所處的階段。這里可以看到,等(ActivityManagerService)AMS啟動完成以后,就可以開始連接Telecom服務(wù)了:

首先,注冊默認(rèn)應(yīng)用(SMS/Dialer etc)通知對象,以便這些應(yīng)用發(fā)送變更(如下載了一個第三方的SMS應(yīng)用時,可以通知系統(tǒng)這一變化); 接著,注冊運(yùn)營商配置變化的廣播接收器,如果配置有變化時,系統(tǒng)會收到通知; 綁定TelecomService,并將其注冊到系統(tǒng)中。

public class TelecomLoaderService extends SystemService { 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'; // 當(dāng)前系統(tǒng)啟動的階段 @Override public void onBootPhase(int phase) { if (phase == PHASE_ACTIVITY_MANAGER_READY) { registerDefaultAppNotifier(); registerCarrierConfigChangedReceiver(); connectToTelecom(); } } //綁定Telecom服務(wù) 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; } } } }

服務(wù)綁定:https://developer.android.com/guide/components/bound-services.html

將服務(wù)添加到ServiceManager中,如果Telecom服務(wù)連接中斷時,則重新連接:

public class TelecomLoaderService extends SystemService { 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') thats redundant here. try { service.linkToDeath(new IBinder.DeathRecipient() { @Override public void binderDied() {connectToTelecom(); } }, 0); SmsApplication.getDefaultMmsApplication(mContext, false); //添加Telecom服務(wù) ServiceManager.addService(Context.TELECOM_SERVICE, service); .... } @Override public void onServiceDisconnected(ComponentName name) { connectToTelecom(); } } }

綁定服務(wù)時,調(diào)用TelecomService的onBind接口,對整個Telecom系統(tǒng)進(jìn)行初始化,并返回一個IBinder接口:

/** * Implementation of the ITelecom interface. */ public class TelecomService extends Service implements TelecomSystem.Component { @Override public IBinder onBind(Intent intent) { // 初始化整個Telecom系統(tǒng) initializeTelecomSystem(this); //返回IBinder接口 synchronized (getTelecomSystem().getLock()) { return getTelecomSystem().getTelecomServiceImpl().getBinder(); } } }

Telecom系統(tǒng)初始化,主要工作是新建一個TelecomSystem的類,在這個類中,會對整個Telecom服務(wù)的相關(guān)類都初始化:

static void initializeTelecomSystem(Context context) { if (TelecomSystem.getInstance() == null) { final NotificationManager notificationManager = (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE); // 用于獲取聯(lián)系人 contactInfoHelper = new ContactInfoHelper(context); // 新建一個單例模式的對象 TelecomSystem.setInstance(new TelecomSystem(....)); } .... } }

構(gòu)造一個單例TelecomSystem對象:

public TelecomSystem( Context context, /* 用戶未接來電通知類(不包括已接或者拒絕的電話) */ MissedCallNotifierImplFactory missedCallNotifierImplFactory, /* 查詢來電信息 */ CallerInfoAsyncQueryFactory callerInfoAsyncQueryFactory, /* 耳機(jī)接入狀態(tài)監(jiān)聽 */ HeadsetMediaButtonFactory headsetMediaButtonFactory, /* 距離傳感器管理 */ ProximitySensorManagerFactory proximitySensorManagerFactory, /* 通話時電話管理 */ InCallWakeLockControllerFactory inCallWakeLockControllerFactory, /* 音頻服務(wù)管理 */ AudioServiceFactory audioServiceFactory, /* 藍(lán)牙設(shè)備管理 */ BluetoothPhoneServiceImplFactory bluetoothPhoneServiceImplFactory, BluetoothVoIPServiceImplFactory bluetoothVoIPServiceImplFactory, /* 查詢所有超時信息 */ Timeouts.Adapter timeoutsAdapter, /* 響鈴播放 */ AsyncRingtonePlayer asyncRingtonePlayer, /* 電話號碼幫助類 */ PhoneNumberUtilsAdapter phoneNumberUtilsAdapter, /* 通話時阻斷通知 */ InterruptionFilterProxy interruptionFilterProxy) { mContext = context.getApplicationContext(); // 初始化telecom相關(guān)的feature TelecomFeature.makeFeature(mContext); // 初始化telecom的數(shù)據(jù)庫 TelecomSystemDB.initialize(mContext); // 創(chuàng)建一個PhoneAccount注冊管理類 mPhoneAccountRegistrar = new PhoneAccountRegistrar(mContext); .... // 初始化通話管家,正是它負(fù)責(zé)與上層UI的交互 mCallsManager = new CallsManager( mContext, mLock, mContactsAsyncHelper, callerInfoAsyncQueryFactory, mMissedCallNotifier, mPhoneAccountRegistrar, headsetMediaButtonFactory, proximitySensorManagerFactory, inCallWakeLockControllerFactory, audioServiceFactory, bluetoothManager, wiredHeadsetManager, systemStateProvider, defaultDialerAdapter, timeoutsAdapter,AsyncRingtonePlayer, phoneNumberUtilsAdapter, interruptionFilterProxy); CallsManager.initialize(mCallsManager); // 注冊需要接收的廣播 mContext.registerReceiver(mUserSwitchedReceiver, USER_SWITCHED_FILTER); mContext.registerReceiver(mUserStartingReceiver, USER_STARTING_FILTER); mContext.registerReceiver(mFeatureChangedReceiver, FEATURE_CHANGED_FILTER); mContext.registerReceiver(mEmergencyReceiver, EMERGENCY_STATE_CHANGED); .... // 所有來電與去電的處理中轉(zhuǎn)站 mCallIntentProcessor = new CallIntentProcessor(mContext, mCallsManager); // 創(chuàng)建一個TelecomServiceImpl用于調(diào)用TelecomService的接口 mTelecomServiceImpl = new TelecomServiceImpl( mContext, mCallsManager, mPhoneAccountRegistrar, new CallIntentProcessor.AdapterImpl(), new UserCallIntentProcessorFactory() { @Override public UserCallIntentProcessor create(Context context, UserHandle userHandle) { return new UserCallIntentProcessor(context, userHandle); } }, defaultDialerAdapter, new TelecomServiceImpl.SubscriptionManagerAdapterImpl(), mLock); // 執(zhí)行特定的初始化操作 initialize(mContext); } }

Android Telephony中的PhoneAccount到底起到個什么作用了?按照源碼中的說明來理解,PhoneAccount表示了不同的接聽或者撥打電話的方式,比如用戶可以通過SIM卡來撥打電話,也可以撥打視頻電話,抑或一個緊急通話,甚至可以通過telephony內(nèi)部的接口來實現(xiàn)撥號,而Android正是通過PhoneAccount來區(qū)分這幾種通話方式的。與之相對應(yīng)的一個類PhoneAccountHandle則是用于表示哪一個用戶正在使用通話服務(wù)。

至此整個Telecom服務(wù)就啟動完成了,這樣Telecom服務(wù)就可以處理來電或者去電了。在接下來的一篇文章里,將分析下來電是如何在Telecom中傳遞與處理,然后發(fā)送到上層UI界面的。

到此這篇關(guān)于Android 系統(tǒng)服務(wù)TelecomService啟動過程原理分析的文章就介紹到這了,更多相關(guān)Android 系統(tǒng)服務(wù)TelecomService啟動內(nèi)容請搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)! $.get('https://blog.csdn.net/wang2119/article/uvc/58164251');
標(biāo)簽: Android
相關(guān)文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
亚洲啊v在线免费视频| 亚洲一级网站| 在线日韩欧美| 黄色日韩在线| 国产99久久| 日韩一级不卡| 亚洲精品一二| 欧美激情麻豆| 国产成年精品| 亚洲二区在线| 涩涩涩久久久成人精品| 国产极品一区| 少妇精品导航| 日韩精品一二三| 国产精品一区高清| 欧美freesex黑人又粗又大| 激情六月综合| 亚洲欧美不卡| 国产精品羞羞答答在线观看| 欧美亚洲tv| 精品91福利视频| 亚洲手机在线| 欧美亚洲福利| 在线观看精品| 蜜桃91丨九色丨蝌蚪91桃色| 国产精品毛片aⅴ一区二区三区| 欧美好骚综合网| 久久国产高清| 九九99久久精品在免费线bt| 蜜臀久久99精品久久一区二区| 亚洲精品日本| 日韩一区亚洲二区| 欧美日韩四区| 国产精品视频一区视频二区| 亚洲高清成人| 国产精品nxnn| 六月婷婷一区| 理论片午夜视频在线观看| 欧美成人午夜| 国产精品久久久久久久免费软件| 亚洲国内精品| 另类欧美日韩国产在线| 久久成人精品| 亚洲精品在线影院| 国产精品最新| 在线精品亚洲| 四虎影视精品| 国产精品亚洲人成在99www| 午夜影院欧美| 91成人网在线观看| 免费亚洲一区| 日本一区二区中文字幕| 久久国产电影| 久久国产精品美女| 综合激情网站| 亚洲婷婷在线| 国产精品成久久久久| 日韩av电影一区| 99成人在线| 精品免费av在线| 久久不卡日韩美女| 午夜久久av| 欧美成人午夜| 久久久久久久久丰满| 欧美黑人做爰爽爽爽| 综合一区在线| 麻豆久久精品| 黄色成人精品网站| 欧美sm一区| 久久精品国产亚洲一区二区三区| 国产一级一区二区| 日韩精品欧美| 国产专区精品| 日本精品国产| 亚洲欧美日本国产专区一区| 三级小说欧洲区亚洲区| 美女视频一区在线观看| 亚洲精品乱码久久久久久蜜桃麻豆| 欧美日韩在线网站| 亚洲精品一区三区三区在线观看| 欧美激情国产在线| av中文字幕在线观看第一页| 欧美激情福利| 国产视频网站一区二区三区| 综合激情网...| 日韩中文字幕1| 久久亚洲影院| 免播放器亚洲| 最近国产精品视频| 亚洲综合小说| 日韩欧美中文字幕一区二区三区 | 最新国产精品视频| 在线观看亚洲精品福利片| 国产色综合网| 日韩中文字幕区一区有砖一区 | 欧美成人亚洲| 夜夜精品视频| 国产麻豆综合| 中文字幕av一区二区三区人| 视频精品一区| 国产精品99精品一区二区三区∴| 久久不见久久见免费视频7| 久久av超碰| 欧美久久天堂| 久久久久99| 亚洲专区一区| 亚洲精品日韩久久| 国产乱码精品一区二区三区四区 | 久久伊人国产| 成人在线视频区| 99久久久久| 午夜一级在线看亚洲| 蜜臀久久99精品久久久画质超高清 | 久久不射中文字幕| 亚洲精品四区| 国产精品va| 精精国产xxxx视频在线野外| 激情欧美一区二区三区| 蜜桃av一区二区在线观看| 日本亚洲视频| 另类欧美日韩国产在线| 久久久久久久久丰满| 亚洲日产国产精品| 欧美一区二区三区久久| 色乱码一区二区三区网站| 欧美a级片一区| 日韩av字幕| 91亚洲成人| 亚洲免费一区二区| 欧美激情三区| 午夜久久免费观看| 日韩国产精品久久久| 国产福利电影在线播放| 欧美专区在线| 国语精品一区| 中文精品电影| 福利在线一区| 国产精品婷婷| 国产精品久久免费视频| 国产一区观看| 国产伦精品一区二区三区视频 | 国际精品欧美精品| 少妇精品导航| 久久精品99国产精品日本| 美女网站视频一区| 亚洲v天堂v手机在线| 久草免费在线视频| 日韩午夜视频在线| 久久中文字幕二区| 免费亚洲一区| 日韩一级不卡| 成人在线视频区| 色综合视频一区二区三区日韩 | 国产欧美综合一区二区三区| 国产在线|日韩| 欧美久久亚洲| 亚洲精品va| 红杏一区二区三区| 日韩欧美中文字幕在线视频| 久久精品动漫| 欧美激情精品| 亚州精品视频| 91九色精品| 久久香蕉网站| 日韩中文av| 亚洲激情黄色| 日韩国产一区二区三区| 欧美日韩一区自拍| 激情91久久| 桃色av一区二区| 国产精品magnet| 蜜臀av亚洲一区中文字幕| 成人在线视频中文字幕| 亚洲tv在线| 老鸭窝亚洲一区二区三区| 日本精品在线中文字幕| 国产精品一区2区3区| 亚洲三级毛片| 欧美午夜不卡| 久久精品国产68国产精品亚洲| 久久精品五月| 欧美激情三区| 色8久久久久| 美女国产一区| 黄色日韩精品| 久久久久久久久丰满| 日韩成人精品一区二区| 国产乱码精品一区二区三区四区 | 欧美91在线| 亚洲黄色影院| 欧美日韩一二三四| 久久久久91| 久久久久欧美精品| 电影天堂国产精品| 日韩在线高清| 欧美一区二区三区激情视频| 九色porny丨国产首页在线| 水蜜桃久久夜色精品一区| 久久中文在线| 乱一区二区av|