1 /*
2  * Copyright (C) 2008 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;
18 
19 import android.app.PendingIntent;
20 import android.app.ProgressDialog;
21 import android.content.BroadcastReceiver;
22 import android.content.Context;
23 import android.content.Intent;
24 import android.os.AsyncTask;
25 import android.os.RecoverySystem;
26 import android.os.storage.StorageManager;
27 import android.provider.Settings;
28 import android.telephony.euicc.EuiccManager;
29 import android.util.Log;
30 import android.util.Slog;
31 import android.view.WindowManager;
32 
33 import com.android.internal.R;
34 
35 import java.io.IOException;
36 import java.util.concurrent.CountDownLatch;
37 import java.util.concurrent.TimeUnit;
38 
39 public class MasterClearReceiver extends BroadcastReceiver {
40     private static final String TAG = "MasterClear";
41     private boolean mWipeExternalStorage;
42     private boolean mWipeEsims;
43 
44     @Override
onReceive(final Context context, final Intent intent)45     public void onReceive(final Context context, final Intent intent) {
46         if (intent.getAction().equals(Intent.ACTION_REMOTE_INTENT)) {
47             if (!"google.com".equals(intent.getStringExtra("from"))) {
48                 Slog.w(TAG, "Ignoring master clear request -- not from trusted server.");
49                 return;
50             }
51         }
52         if (Intent.ACTION_MASTER_CLEAR.equals(intent.getAction())) {
53             Slog.w(TAG, "The request uses the deprecated Intent#ACTION_MASTER_CLEAR, "
54                     + "Intent#ACTION_FACTORY_RESET should be used instead.");
55         }
56         if (intent.hasExtra(Intent.EXTRA_FORCE_MASTER_CLEAR)) {
57             Slog.w(TAG, "The request uses the deprecated Intent#EXTRA_FORCE_MASTER_CLEAR, "
58                     + "Intent#EXTRA_FORCE_FACTORY_RESET should be used instead.");
59         }
60 
61         final boolean shutdown = intent.getBooleanExtra("shutdown", false);
62         final String reason = intent.getStringExtra(Intent.EXTRA_REASON);
63         mWipeExternalStorage = intent.getBooleanExtra(Intent.EXTRA_WIPE_EXTERNAL_STORAGE, false);
64         mWipeEsims = intent.getBooleanExtra(Intent.EXTRA_WIPE_ESIMS, false);
65         final boolean forceWipe = intent.getBooleanExtra(Intent.EXTRA_FORCE_MASTER_CLEAR, false)
66                 || intent.getBooleanExtra(Intent.EXTRA_FORCE_FACTORY_RESET, false);
67 
68         Slog.w(TAG, "!!! FACTORY RESET !!!");
69         // The reboot call is blocking, so we need to do it on another thread.
70         Thread thr = new Thread("Reboot") {
71             @Override
72             public void run() {
73                 try {
74                     RecoverySystem
75                             .rebootWipeUserData(context, shutdown, reason, forceWipe, mWipeEsims);
76                     Log.wtf(TAG, "Still running after master clear?!");
77                 } catch (IOException e) {
78                     Slog.e(TAG, "Can't perform master clear/factory reset", e);
79                 } catch (SecurityException e) {
80                     Slog.e(TAG, "Can't perform master clear/factory reset", e);
81                 }
82             }
83         };
84 
85         if (mWipeExternalStorage || mWipeEsims) {
86             // thr will be started at the end of this task.
87             new WipeDataTask(context, thr).execute();
88         } else {
89             thr.start();
90         }
91     }
92 
93     private class WipeDataTask extends AsyncTask<Void, Void, Void> {
94         private final Thread mChainedTask;
95         private final Context mContext;
96         private final ProgressDialog mProgressDialog;
97 
WipeDataTask(Context context, Thread chainedTask)98         public WipeDataTask(Context context, Thread chainedTask) {
99             mContext = context;
100             mChainedTask = chainedTask;
101             mProgressDialog = new ProgressDialog(context);
102         }
103 
104         @Override
onPreExecute()105         protected void onPreExecute() {
106             mProgressDialog.setIndeterminate(true);
107             mProgressDialog.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);
108             mProgressDialog.setMessage(mContext.getText(R.string.progress_erasing));
109             mProgressDialog.show();
110         }
111 
112         @Override
doInBackground(Void... params)113         protected Void doInBackground(Void... params) {
114             Slog.w(TAG, "Wiping adoptable disks");
115             if (mWipeExternalStorage) {
116                 StorageManager sm = (StorageManager) mContext.getSystemService(
117                         Context.STORAGE_SERVICE);
118                 sm.wipeAdoptableDisks();
119             }
120             return null;
121         }
122 
123         @Override
onPostExecute(Void result)124         protected void onPostExecute(Void result) {
125             mProgressDialog.dismiss();
126             mChainedTask.start();
127         }
128 
129     }
130 }
131