1 /* 2 * Copyright (C) 2016 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License 15 */ 16 17 package com.android.server.am; 18 19 import static android.content.pm.PackageManager.MATCH_SYSTEM_ONLY; 20 21 import android.app.AppOpsManager; 22 import android.app.Notification; 23 import android.app.NotificationManager; 24 import android.app.PendingIntent; 25 import android.content.ComponentName; 26 import android.content.Context; 27 import android.content.IIntentReceiver; 28 import android.content.Intent; 29 import android.content.pm.ResolveInfo; 30 import android.os.Binder; 31 import android.os.Bundle; 32 import android.os.Handler; 33 import android.os.Message; 34 import android.os.Process; 35 import android.os.UserHandle; 36 import android.util.Slog; 37 38 import com.android.internal.R; 39 import com.android.internal.messages.nano.SystemMessageProto.SystemMessage; 40 import com.android.internal.notification.SystemNotificationChannels; 41 import com.android.internal.util.ProgressReporter; 42 import com.android.server.UiThread; 43 44 import java.util.List; 45 46 /** 47 * Simple broadcaster that sends {@link Intent#ACTION_PRE_BOOT_COMPLETED} to all 48 * system apps that register for it. Override {@link #onFinished()} to handle 49 * when all broadcasts are finished. 50 */ 51 public abstract class PreBootBroadcaster extends IIntentReceiver.Stub { 52 private static final String TAG = "PreBootBroadcaster"; 53 54 private final ActivityManagerService mService; 55 private final int mUserId; 56 private final ProgressReporter mProgress; 57 private final boolean mQuiet; 58 59 private final Intent mIntent; 60 private final List<ResolveInfo> mTargets; 61 62 private int mIndex = 0; 63 PreBootBroadcaster(ActivityManagerService service, int userId, ProgressReporter progress, boolean quiet)64 public PreBootBroadcaster(ActivityManagerService service, int userId, 65 ProgressReporter progress, boolean quiet) { 66 mService = service; 67 mUserId = userId; 68 mProgress = progress; 69 mQuiet = quiet; 70 71 mIntent = new Intent(Intent.ACTION_PRE_BOOT_COMPLETED); 72 mIntent.addFlags(Intent.FLAG_RECEIVER_BOOT_UPGRADE | Intent.FLAG_DEBUG_TRIAGED_MISSING); 73 74 mTargets = mService.mContext.getPackageManager().queryBroadcastReceiversAsUser(mIntent, 75 MATCH_SYSTEM_ONLY, UserHandle.of(userId)); 76 } 77 sendNext()78 public void sendNext() { 79 if (mIndex >= mTargets.size()) { 80 mHandler.obtainMessage(MSG_HIDE).sendToTarget(); 81 onFinished(); 82 return; 83 } 84 85 if (!mService.isUserRunning(mUserId, 0)) { 86 Slog.i(TAG, "User " + mUserId + " is no longer running; skipping remaining receivers"); 87 mHandler.obtainMessage(MSG_HIDE).sendToTarget(); 88 onFinished(); 89 return; 90 } 91 92 if (!mQuiet) { 93 mHandler.obtainMessage(MSG_SHOW, mTargets.size(), mIndex).sendToTarget(); 94 } 95 96 final ResolveInfo ri = mTargets.get(mIndex++); 97 final ComponentName componentName = ri.activityInfo.getComponentName(); 98 99 if (mProgress != null) { 100 final CharSequence label = ri.activityInfo 101 .loadLabel(mService.mContext.getPackageManager()); 102 mProgress.setProgress(mIndex, mTargets.size(), 103 mService.mContext.getString(R.string.android_preparing_apk, label)); 104 } 105 106 Slog.i(TAG, "Pre-boot of " + componentName.toShortString() + " for user " + mUserId); 107 EventLogTags.writeAmPreBoot(mUserId, componentName.getPackageName()); 108 109 mIntent.setComponent(componentName); 110 synchronized (mService) { 111 mService.broadcastIntentLocked(null, null, null, mIntent, null, this, 0, null, null, 112 null, AppOpsManager.OP_NONE, null, true, false, ActivityManagerService.MY_PID, 113 Process.SYSTEM_UID, Binder.getCallingUid(), Binder.getCallingPid(), mUserId); 114 } 115 } 116 117 @Override performReceive(Intent intent, int resultCode, String data, Bundle extras, boolean ordered, boolean sticky, int sendingUser)118 public void performReceive(Intent intent, int resultCode, String data, Bundle extras, 119 boolean ordered, boolean sticky, int sendingUser) { 120 sendNext(); 121 } 122 123 private static final int MSG_SHOW = 1; 124 private static final int MSG_HIDE = 2; 125 126 private Handler mHandler = new Handler(UiThread.get().getLooper(), null, true) { 127 @Override 128 public void handleMessage(Message msg) { 129 final Context context = mService.mContext; 130 final NotificationManager notifManager = context 131 .getSystemService(NotificationManager.class); 132 final int max = msg.arg1; 133 final int index = msg.arg2; 134 135 switch (msg.what) { 136 case MSG_SHOW: 137 final CharSequence title = context 138 .getText(R.string.android_upgrading_notification_title); 139 140 final Intent intent = new Intent(); 141 intent.setClassName("com.android.settings", 142 "com.android.settings.HelpTrampoline"); 143 intent.putExtra(Intent.EXTRA_TEXT, "help_url_upgrading"); 144 145 final PendingIntent contentIntent; 146 if (context.getPackageManager().resolveActivity(intent, 0) != null) { 147 contentIntent = PendingIntent.getActivity(context, 0, intent, 0); 148 } else { 149 contentIntent = null; 150 } 151 152 final Notification notif = 153 new Notification.Builder(mService.mContext, 154 SystemNotificationChannels.UPDATES) 155 .setSmallIcon(R.drawable.stat_sys_adb) 156 .setWhen(0) 157 .setOngoing(true) 158 .setTicker(title) 159 .setColor(context.getColor( 160 com.android.internal.R.color.system_notification_accent_color)) 161 .setContentTitle(title) 162 .setContentIntent(contentIntent) 163 .setVisibility(Notification.VISIBILITY_PUBLIC) 164 .setProgress(max, index, false) 165 .build(); 166 notifManager.notifyAsUser(TAG, SystemMessage.NOTE_SYSTEM_UPGRADING, notif, 167 UserHandle.of(mUserId)); 168 break; 169 170 case MSG_HIDE: 171 notifManager.cancelAsUser(TAG, SystemMessage.NOTE_SYSTEM_UPGRADING, 172 UserHandle.of(mUserId)); 173 break; 174 } 175 } 176 }; 177 onFinished()178 public abstract void onFinished(); 179 } 180