搜档网
当前位置:搜档网 › Android的时间更新

Android的时间更新

Android的时间更新
Android的时间更新

Android的时间更新分成2种,一种是走运营商协议的NITZ,另外一种是走网络时钟的SNTP。

1.SNTP:

SNTP的逻辑主要分布在NetworkTimeUpdateService。它通过监听ACTION_NETWORK_SET_TIME,ACTION_NETWORK_SET_TIMEZONE这两个事件来判断最近一段时间内是否有NITZ的时间已经被更新过;它通过监听ConnectivityManager的触发事件来判断WIFI网络的连接,以此来触发网络事件的更新。

PS:1. NtpTrustedTime class来真正的实施从网络侧获取SNTP的时间。2.AlarmManager来实施定时重查机制。

2.NITZ:根据代码,NITZ需要运营商支持,通过TOD时间信息同步法进行时间同步。TOD,

是指于“短波跳频通信”的一种基于精确时钟的同步法。“时间信息”包含了跳频的状态信息和时间信息,状态信息是指伪随机码(PN)发生器的实时码序列状态;时间信息是指实时时钟信息,即年、月、日、时、分、秒、毫秒、微秒、毫微秒等的精确时间。根据这些信息,就实现运营商网络侧和手机modem侧的时间同步。

3.

Qualcomm 在android系统bootup的时候启动一个time_daemon的守护进程。目前米2的qualcomm的实现是通过time_daemon的进程,来维护一个从modem获取时间,更新RTC的方法。

第一部分,Time_daemon的初始化以及其和modem的时间同步

Time_daemon初始化:

第一步,首先通过genoff_init_config来初始化一些基本设置。

这个初始化做了以下2件事情,其通过ats_rtc_init来初始化ats_bases[0],也就是和RTC 相关的内容。具体做法是从/dev/rtc0中来获取RTC的值,并转换成UTC的milliseconds 之后,将其存入到time_genoff_ptr这个结构的generic_offset里面去。最后通过genoff_post_init这个函数初始化了time_genoff_ptr这个结构

通过ats_bases_init初始化了ats_bases[ATS_MAX]里面其他的值。

第二部,通过genoff_boot_tod_init来初始化time_genoff_info_type的结构

第三步,通过genoff_modem_qmi_init来初始化QMI的service和client,它通过time_get_service_object_v01来获取modem的service,用qmi_client_notifier_init(),qmi_client_get_service_list(),qmi_client_init(),来建立一个采取qmi通信机制的,可被触发的client。并获取时间初始值gettimeofday(),并通过qmi_client_send_msg_sync ()来发送给modem。

第四步,通过pthread_create和pthread_join来维持一个和modem同步的socket

connection。(conn_handler,read_offset都作为函数指针作为参数传入)

初始化完成之后

Read_offset()通过qmi_client_send_msg_sync来从modem测获取generic_offset的值,并通过genoff_opr来写入generic_offset的值(TODO:MODEM_EPOCH_DIFFERENCE

315964800????)

几个结构,ats_bases[0]存放了RTC的相关信息,ats_base是一个最大值为ATS_MAX

的存放了time_genoff_struct结构的数组。

time_genoff_info_type:Structure to be passed as argument to time_genoff_operation(),其中存放了时间(time_unit),操作(time_genoff_opr)

第二部分:APP如何被动的被触发更新NITZ

APP最终通过time_genoff_operation()函数来open一个socket与time daemon通信,从而从modem测获取时间数据.

qcril_cm_process_network_info()会在一定条件下被触发(qcril_cm_event_card_status_updated里被调用,其会在收到modem的消息时被触发执行),在qcril_cm_process_network_info()里面,qcril_cm_prep_nitz_time_received_report ()一旦执行成功,会调用qcril_default_unsol_resp_params()来发送RIL_UNSOL_NITZ_TIME_RECEIVED消息,RIL_UNSOL_NITZ_TIME_RECEIVED会被RIL接受到触发一个mNITZTimeRegistrant的notifyRegistrant。

CdmaServiceStateTracker()在启动的时候通过setOnNITZTime向RIL注册,从而在这个时候能被notifyRegistrant()触发,从而执行setTimeFromNITZString(),最终在此刻,计算出当前的时间,并通过SystemClock.setCurrentTimeMillis()写入系统,同时广播一个ACTION_NETWORK_SET_TIME消息。

PS: APP 测的代码主要分布在

framework/base/services/java/android/server --NetworkTimeUpdateService framework/base/core/java/android/util ---NtpTrustedTime

framework/base/telephony/java/com/android/internal/telephony/cdmaCdmaServiceStateTr acker

其他部分主要分布在vendor/qcom/proprietary/time-services

TODO:1.需要更好的理解QMI的机制从而更深入的理解Time_daemon从modem update时间的过程。2.更为清晰的流程图3.Modem侧是通过什么方式与运营商通信获取时间的?

相关主题