android平台预置失效解决

Android平台预置GMS包后关机闹钟失效问题及解决方法

编程开发 2020-10-16 08:07:07 38

导读

1.介绍关机闹钟为Android中默认支持的功能,实现起来则需要满足一定的条件:自动开机、开机后响铃。对于自动开机来说,自动关机可以在应用层通过设置alarm来实现,而自动开机需要底层rtc时钟的支持;开机后检查时间,到点响铃。2.原理一般智能手机的硬件架构都是分为RF,BB,……

1.介绍

关机闹钟为Android中默认支持的功能,实现起来则需要满足一定的条件:自动开机、开机后响铃。对于自动开机来说,自动关机可以在应用层通过设置alarm来实现,而自动开机需要底层rtc时钟的支持;开机后检查时间,到点响铃。

2.原理

一般智能手机的硬件架构都是分为RF,BB,AP这三个部分, RF射频部分的功能主要是接收和发射射频,大家不用关注这个。而AP部分就是应用程序处理器,其实也就是CPU模块;BB部分则是基带模块。通常手机关机都是AP模块完全断电,而BB虽然处在关机状态,但实际上闹钟和时间部分并没有断电。而当所设置闹钟到点时BB模块会自动给AP模块上电,这样系统也就能开机并激活闹钟功能了。

3.流程

在闹钟设置一个时间合适且有效响铃的闹钟后,向PowerOffAlarm发送设定关机闹钟广播并传入闹钟时间参数,PowerOffAlarm接收到广播后,根据预设提前开机时间和闹钟时间往rtc中写入时间,并将该时间写入文件中暂存。
设置好闹钟后关机,机器会根据rtc中的alrm_time时间参数来开机,开机后AlarmManagerService会检查闹钟,并在闹钟时间发送广播后弹出响铃界面。
点击关闭闹钟后,会发送取消关机闹钟广播,PowerOffAlarm会对传入时间与暂存的时间对比,如果相同则会取消该关机闹钟。

4.遇到问题及解决方法

4.1预置GMS包后,关机闹钟失效

在对比平台闹钟分析后,发现谷歌闹钟在设置闹钟后并没有向PowerOffAlarm发送广播,但是由于谷歌闹钟无法进行修改。
在分析了平台的闹钟设置闹钟步骤后,发现在设置闹钟后都会调用AlarmManager.setAlarmClock()这个方法,然后再AlarmStateManager的updateNextAlarm方法中发送设置关机闹钟的广播。
谷歌闹钟在设置闹钟也会调用AlarmManager.setAlarmClock()这个方法,所以在该方法中直接发送广播到PowerOffAlarm中。

@@ -615,8 +633,34 @@ public class AlarmManager {
  public void setAlarmClock(AlarmClockInfo info, PendingIntent operation) {
   setImpl(RTC_WAKEUP, info.getTriggerTime(), WINDOW_EXACT, 0, 0, operation,
     null, null, null, null, info);
+  setPowerOffAlarm(mContext,info.getTriggerTime());
  }
 
+
+ /*add by hxj at Aug.6,for PowerOffAlarm*/
+ private void setPowerOffAlarm(Context context, long time) {
+  Log.d(TAG, "setPowerOffAlarm: saveAlarmToPreference and sendBroadcast to setPowerOffAlarm,the time is "+time);
+  DEFAULT_ALARM_TIME = time;
+  Intent intent = new Intent(ACTION_SET_POWEROFF_ALARM);
+  intent.addFlags(Intent.FLAG_RECEIVER_FOREGROUND);
+  intent.setPackage(POWER_OFF_ALARM_PACKAGE);
+  intent.putExtra(TIME, time);
+  context.sendBroadcast(intent);
+ }
+
+ private void cancelPowerOffAlarm(Context context) {
+  //long time = getAlarmFromPreference(context);
+  Log.d(TAG, "cancelPowerOffAlarm: cancel alarm and cancelPowerOffAlarm(mContext),time is "+DEFAULT_ALARM_TIME );
+  Intent intent = new Intent(ACTION_CANCEL_POWEROFF_ALARM);
+  intent.addFlags(Intent.FLAG_RECEIVER_FOREGROUND);
+  intent.putExtra(TIME, DEFAULT_ALARM_TIME);
+  intent.setPackage(POWER_OFF_ALARM_PACKAGE);
+  context.sendBroadcast(intent);
+  DEFAULT_ALARM_TIME = 0L;
+ }
+
+
  /** @hide */
  @SystemApi
  @RequiresPermission(android.Manifest.permission.UPDATE_DEVICE_STATS)
@@ -930,6 +974,7 @@ public class AlarmManager {
 
   try {
    mService.remove(operation, null);
+   cancelPowerOffAlarm(mContext);
   } catch (RemoteException ex) {
    throw ex.rethrowFromSystemServer();
   }

4.2 PowerOffAlarm无法接收广播

在分析log后,发现接受该广播需要在应用中添加特定的权限,即"org.codeaurora.permission.POWER_OFF_ALARM"权限,但是由于谷歌闹钟中未添加并且无法修改,所以在PowerOffAlarm中接收管广播处移除该权限。

@@ -29,7 +29,7 @@
   android:defaultToDeviceProtectedStorage="true">
 
   <receiver android:name=".PowerOffAlarmBroadcastReceiver"
-   android:permission="org.codeaurora.permission.POWER_OFF_ALARM"
+
    android:exported="true"
    android:directBootAware="true"
    android:label="PowerOffAlarmBroadcastReceiver">

4.3 开机时间过长导致开机后闹钟过期

提前开机时间平台预设默认为90000毫秒,项目中开机时间较长,所以增长开机时间,改为150000毫秒。

@@ -42,7 +42,7 @@ public class PowerOffAlarmUtils {
 
  private static final int FAILURE = -1;
 
- public static final long MS_IN_ONE_MIN = 90000L;
+ public static final long MS_IN_ONE_MIN = 150000L;
 
  private static final long SEC_TO_MS = 1000L;

此外还有其他的问题也会导致关机闹钟的失效,有时间再来记录一下。


1253067 TFnetwork_cn