CryptKeeper.java revision f0104df823a93ef4c3336118ddb02a0924e520da
1/*
2 * Copyright (C) 2011 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
17package com.android.settings;
18
19import android.app.Activity;
20import android.app.StatusBarManager;
21import android.content.ComponentName;
22import android.content.Context;
23import android.content.Intent;
24import android.content.pm.PackageManager;
25import android.graphics.Rect;
26import android.inputmethodservice.KeyboardView;
27import android.os.AsyncTask;
28import android.os.Bundle;
29import android.os.Handler;
30import android.os.IBinder;
31import android.os.Message;
32import android.os.PowerManager;
33import android.os.RemoteException;
34import android.os.ServiceManager;
35import android.os.SystemProperties;
36import android.os.storage.IMountService;
37import android.telephony.TelephonyManager;
38import android.text.TextUtils;
39import android.util.AttributeSet;
40import android.util.Log;
41import android.view.KeyEvent;
42import android.view.MotionEvent;
43import android.view.View;
44import android.view.View.OnClickListener;
45import android.view.inputmethod.EditorInfo;
46import android.view.inputmethod.InputMethodManager;
47import android.widget.Button;
48import android.widget.EditText;
49import android.widget.ProgressBar;
50import android.widget.TextView;
51
52import com.android.internal.telephony.ITelephony;
53import com.android.internal.widget.PasswordEntryKeyboardHelper;
54import com.android.internal.widget.PasswordEntryKeyboardView;
55
56/**
57 * Settings screens to show the UI flows for encrypting/decrypting the device.
58 *
59 * This may be started via adb for debugging the UI layout, without having to go through
60 * encryption flows everytime. It should be noted that starting the activity in this manner
61 * is only useful for verifying UI-correctness - the behavior will not be identical.
62 * <pre>
63 * $ adb shell pm enable com.android.settings/.CryptKeeper
64 * $ adb shell am start \
65 *     -e "com.android.settings.CryptKeeper.DEBUG_FORCE_VIEW" "progress" \
66 *     -n com.android.settings/.CryptKeeper
67 * </pre>
68 */
69public class CryptKeeper extends Activity implements TextView.OnEditorActionListener {
70    private static final String TAG = "CryptKeeper";
71
72    private static final String DECRYPT_STATE = "trigger_restart_framework";
73
74    private static final int UPDATE_PROGRESS = 1;
75    private static final int COOLDOWN = 2;
76
77    private static final int MAX_FAILED_ATTEMPTS = 30;
78    private static final int COOL_DOWN_ATTEMPTS = 10;
79    private static final int COOL_DOWN_INTERVAL = 30; // 30 seconds
80
81    // Intent action for launching the Emergency Dialer activity.
82    static final String ACTION_EMERGENCY_DIAL = "com.android.phone.EmergencyDialer.DIAL";
83
84    // Debug Intent extras so that this Activity may be started via adb for debugging UI layouts
85    private static final String EXTRA_FORCE_VIEW =
86            "com.android.settings.CryptKeeper.DEBUG_FORCE_VIEW";
87    private static final String FORCE_VIEW_PROGRESS = "progress";
88    private static final String FORCE_VIEW_ENTRY = "entry";
89    private static final String FORCE_VIEW_ERROR = "error";
90
91    private int mCooldown;
92    PowerManager.WakeLock mWakeLock;
93    private EditText mPasswordEntry;
94
95    /**
96     * Used to propagate state through configuration changes (e.g. screen rotation)
97     */
98    private static class NonConfigurationInstanceState {
99        final PowerManager.WakeLock wakelock;
100
101        NonConfigurationInstanceState(PowerManager.WakeLock _wakelock) {
102            wakelock = _wakelock;
103        }
104    }
105
106    // This activity is used to fade the screen to black after the password is entered.
107    public static class Blank extends Activity {
108        @Override
109        public void onCreate(Bundle savedInstanceState) {
110            super.onCreate(savedInstanceState);
111            setContentView(R.layout.crypt_keeper_blank);
112        }
113    }
114
115    // Use a custom EditText to prevent the input method from showing.
116    public static class CryptEditText extends EditText {
117        InputMethodManager imm;
118
119        public CryptEditText(Context context, AttributeSet attrs) {
120            super(context, attrs);
121            imm = ((InputMethodManager) getContext().
122                    getSystemService(Context.INPUT_METHOD_SERVICE));
123        }
124
125        @Override
126        protected void onFocusChanged(boolean focused, int direction, Rect previouslyFocusedRect) {
127            super.onFocusChanged(focused, direction, previouslyFocusedRect);
128
129            if (focused && imm != null && imm.isActive(this)) {
130                imm.hideSoftInputFromWindow(getApplicationWindowToken(), 0);
131            }
132        }
133
134        @Override
135        public boolean onTouchEvent(MotionEvent event) {
136            boolean handled = super.onTouchEvent(event);
137
138            if (imm != null && imm.isActive(this)) {
139                imm.hideSoftInputFromWindow(getApplicationWindowToken(), 0);
140            }
141
142            return handled;
143        }
144    }
145
146    private class DecryptTask extends AsyncTask<String, Void, Integer> {
147        @Override
148        protected Integer doInBackground(String... params) {
149            IMountService service = getMountService();
150            try {
151                return service.decryptStorage(params[0]);
152            } catch (Exception e) {
153                Log.e(TAG, "Error while decrypting...", e);
154                return -1;
155            }
156        }
157
158        @Override
159        protected void onPostExecute(Integer failedAttempts) {
160            if (failedAttempts == 0) {
161                // The password was entered successfully. Start the Blank activity
162                // so this activity animates to black before the devices starts. Note
163                // It has 1 second to complete the animation or it will be frozen
164                // until the boot animation comes back up.
165                Intent intent = new Intent(CryptKeeper.this, Blank.class);
166                finish();
167                startActivity(intent);
168            } else if (failedAttempts == MAX_FAILED_ATTEMPTS) {
169                // Factory reset the device.
170                sendBroadcast(new Intent("android.intent.action.MASTER_CLEAR"));
171            } else if ((failedAttempts % COOL_DOWN_ATTEMPTS) == 0) {
172                mCooldown = COOL_DOWN_INTERVAL;
173                cooldown();
174            } else {
175                TextView tv = (TextView) findViewById(R.id.status);
176                tv.setText(R.string.try_again);
177                tv.setVisibility(View.VISIBLE);
178
179                // Reenable the password entry
180                mPasswordEntry.setEnabled(true);
181            }
182        }
183    }
184
185    private final Handler mHandler = new Handler() {
186        @Override
187        public void handleMessage(Message msg) {
188            switch (msg.what) {
189            case UPDATE_PROGRESS:
190                updateProgress();
191                break;
192
193            case COOLDOWN:
194                cooldown();
195                break;
196            }
197        }
198    };
199
200    /** @return whether or not this Activity was started for debugging the UI only. */
201    private boolean isDebugView() {
202        return getIntent().hasExtra(EXTRA_FORCE_VIEW);
203    }
204
205    /** @return whether or not this Activity was started for debugging the specific UI view only. */
206    private boolean isDebugView(String viewType /* non-nullable */) {
207        return viewType.equals(getIntent().getStringExtra(EXTRA_FORCE_VIEW));
208    }
209
210    @Override
211    public void onCreate(Bundle savedInstanceState) {
212        super.onCreate(savedInstanceState);
213
214        // If we are not encrypted or encrypting, get out quickly.
215        String state = SystemProperties.get("vold.decrypt");
216        if (!isDebugView() && ("".equals(state) || DECRYPT_STATE.equals(state))) {
217            // Disable the crypt keeper.
218            PackageManager pm = getPackageManager();
219            ComponentName name = new ComponentName(this, CryptKeeper.class);
220            pm.setComponentEnabledSetting(name, PackageManager.COMPONENT_ENABLED_STATE_DISABLED, 0);
221            return;
222        }
223
224        // Disable the status bar
225        StatusBarManager sbm = (StatusBarManager) getSystemService(Context.STATUS_BAR_SERVICE);
226        sbm.disable(StatusBarManager.DISABLE_EXPAND
227                | StatusBarManager.DISABLE_NOTIFICATION_ICONS
228                | StatusBarManager.DISABLE_NOTIFICATION_ALERTS
229                | StatusBarManager.DISABLE_SYSTEM_INFO
230                | StatusBarManager.DISABLE_NAVIGATION
231                | StatusBarManager.DISABLE_BACK);
232
233        // Check for (and recover) retained instance data
234        Object lastInstance = getLastNonConfigurationInstance();
235        if (lastInstance instanceof NonConfigurationInstanceState) {
236            NonConfigurationInstanceState retained = (NonConfigurationInstanceState) lastInstance;
237            mWakeLock = retained.wakelock;
238        }
239    }
240
241    /**
242     * Note, we defer the state check and screen setup to onStart() because this will be
243     * re-run if the user clicks the power button (sleeping/waking the screen), and this is
244     * especially important if we were to lose the wakelock for any reason.
245     */
246    @Override
247    public void onStart() {
248        super.onStart();
249
250        // Check to see why we were started.
251        String progress = SystemProperties.get("vold.encrypt_progress");
252        if (!"".equals(progress)
253                || isDebugView(FORCE_VIEW_PROGRESS)
254                || isDebugView(FORCE_VIEW_ERROR)) {
255            setContentView(R.layout.crypt_keeper_progress);
256            encryptionProgressInit();
257        } else {
258            setContentView(R.layout.crypt_keeper_password_entry);
259            passwordEntryInit();
260        }
261    }
262
263    @Override
264    public void onStop() {
265        super.onStop();
266
267        mHandler.removeMessages(COOLDOWN);
268        mHandler.removeMessages(UPDATE_PROGRESS);
269    }
270
271    /**
272     * Reconfiguring, so propagate the wakelock to the next instance.  This runs between onStop()
273     * and onDestroy() and only if we are changing configuration (e.g. rotation).  Also clears
274     * mWakeLock so the subsequent call to onDestroy does not release it.
275     */
276    @Override
277    public Object onRetainNonConfigurationInstance() {
278        NonConfigurationInstanceState state = new NonConfigurationInstanceState(mWakeLock);
279        mWakeLock = null;
280        return state;
281    }
282
283    @Override
284    public void onDestroy() {
285        super.onDestroy();
286
287        if (mWakeLock != null) {
288            mWakeLock.release();
289            mWakeLock = null;
290        }
291    }
292
293    private void encryptionProgressInit() {
294        // Accquire a partial wakelock to prevent the device from sleeping. Note
295        // we never release this wakelock as we will be restarted after the device
296        // is encrypted.
297
298        PowerManager pm = (PowerManager) getSystemService(Context.POWER_SERVICE);
299        mWakeLock = pm.newWakeLock(PowerManager.FULL_WAKE_LOCK, TAG);
300
301        mWakeLock.acquire();
302
303        ProgressBar progressBar = (ProgressBar) findViewById(R.id.progress_bar);
304        progressBar.setIndeterminate(true);
305
306        updateProgress();
307    }
308
309    private void showFactoryReset() {
310        // Hide the encryption-bot to make room for the "factory reset" button
311        findViewById(R.id.encroid).setVisibility(View.GONE);
312
313        // Show the reset button, failure text, and a divider
314        Button button = (Button) findViewById(R.id.factory_reset);
315        button.setVisibility(View.VISIBLE);
316        button.setOnClickListener(new OnClickListener() {
317            public void onClick(View v) {
318                // Factory reset the device.
319                sendBroadcast(new Intent("android.intent.action.MASTER_CLEAR"));
320            }
321        });
322
323        TextView tv = (TextView) findViewById(R.id.title);
324        tv.setText(R.string.crypt_keeper_failed_title);
325
326        tv = (TextView) findViewById(R.id.status);
327        tv.setText(R.string.crypt_keeper_failed_summary);
328
329        View view = findViewById(R.id.bottom_divider);
330        if (view != null) {
331            view.setVisibility(View.VISIBLE);
332        }
333    }
334
335    private void updateProgress() {
336        String state = SystemProperties.get("vold.encrypt_progress");
337
338        if ("error_partially_encrypted".equals(state) || isDebugView(FORCE_VIEW_ERROR)) {
339            showFactoryReset();
340            return;
341        }
342
343        int progress = 0;
344        try {
345            // Force a 50% progress state when debugging the view.
346            progress = isDebugView() ? 50 : Integer.parseInt(state);
347        } catch (Exception e) {
348            Log.w(TAG, "Error parsing progress: " + e.toString());
349        }
350
351        CharSequence status = getText(R.string.crypt_keeper_setup_description);
352        TextView tv = (TextView) findViewById(R.id.status);
353        tv.setText(TextUtils.expandTemplate(status, Integer.toString(progress)));
354
355        // Check the progress every 5 seconds
356        mHandler.removeMessages(UPDATE_PROGRESS);
357        mHandler.sendEmptyMessageDelayed(UPDATE_PROGRESS, 5000);
358    }
359
360    private void cooldown() {
361        TextView tv = (TextView) findViewById(R.id.status);
362
363        if (mCooldown <= 0) {
364            // Re-enable the password entry
365            mPasswordEntry.setEnabled(true);
366
367            tv.setVisibility(View.GONE);
368        } else {
369            CharSequence template = getText(R.string.crypt_keeper_cooldown);
370            tv.setText(TextUtils.expandTemplate(template, Integer.toString(mCooldown)));
371
372            tv.setVisibility(View.VISIBLE);
373
374            mCooldown--;
375            mHandler.removeMessages(COOLDOWN);
376            mHandler.sendEmptyMessageDelayed(COOLDOWN, 1000); // Tick every second
377        }
378    }
379
380    private void passwordEntryInit() {
381        mPasswordEntry = (EditText) findViewById(R.id.passwordEntry);
382        mPasswordEntry.setOnEditorActionListener(this);
383
384        KeyboardView keyboardView = (PasswordEntryKeyboardView) findViewById(R.id.keyboard);
385
386        if (keyboardView != null) {
387            PasswordEntryKeyboardHelper keyboardHelper = new PasswordEntryKeyboardHelper(this,
388                    keyboardView, mPasswordEntry, false);
389            keyboardHelper.setKeyboardMode(PasswordEntryKeyboardHelper.KEYBOARD_MODE_ALPHA);
390        }
391
392        updateEmergencyCallButtonState();
393    }
394
395    private IMountService getMountService() {
396        IBinder service = ServiceManager.getService("mount");
397        if (service != null) {
398            return IMountService.Stub.asInterface(service);
399        }
400        return null;
401    }
402
403    @Override
404    public boolean onEditorAction(TextView v, int actionId, KeyEvent event) {
405        if (actionId == EditorInfo.IME_NULL || actionId == EditorInfo.IME_ACTION_DONE) {
406            // Get the password
407            String password = v.getText().toString();
408
409            if (TextUtils.isEmpty(password)) {
410                return true;
411            }
412
413            // Now that we have the password clear the password field.
414            v.setText(null);
415
416            // Disable the password entry while checking the password. This
417            // we either be reenabled if the password was wrong or after the
418            // cooldown period.
419            mPasswordEntry.setEnabled(false);
420
421            new DecryptTask().execute(password);
422
423            return true;
424        }
425        return false;
426    }
427
428    //
429    // Code to update the state of, and handle clicks from, the "Emergency call" button.
430    //
431    // This code is mostly duplicated from the corresponding code in
432    // LockPatternUtils and LockPatternKeyguardView under frameworks/base.
433    //
434
435    private void updateEmergencyCallButtonState() {
436        Button button = (Button) findViewById(R.id.emergencyCallButton);
437        // The button isn't present at all in some configurations.
438        if (button == null) return;
439
440        if (isEmergencyCallCapable()) {
441            button.setVisibility(View.VISIBLE);
442            button.setOnClickListener(new View.OnClickListener() {
443                    public void onClick(View v) {
444                        takeEmergencyCallAction();
445                    }
446                });
447        } else {
448            button.setVisibility(View.GONE);
449            return;
450        }
451
452        int newState = TelephonyManager.getDefault().getCallState();
453        int textId;
454        if (newState == TelephonyManager.CALL_STATE_OFFHOOK) {
455            // show "return to call" text and show phone icon
456            textId = R.string.cryptkeeper_return_to_call;
457            int phoneCallIcon = R.drawable.stat_sys_phone_call;
458            button.setCompoundDrawablesWithIntrinsicBounds(phoneCallIcon, 0, 0, 0);
459        } else {
460            textId = R.string.cryptkeeper_emergency_call;
461            int emergencyIcon = R.drawable.ic_emergency;
462            button.setCompoundDrawablesWithIntrinsicBounds(emergencyIcon, 0, 0, 0);
463        }
464        button.setText(textId);
465    }
466
467    private boolean isEmergencyCallCapable() {
468        return getResources().getBoolean(com.android.internal.R.bool.config_voice_capable);
469    }
470
471    private void takeEmergencyCallAction() {
472        if (TelephonyManager.getDefault().getCallState() == TelephonyManager.CALL_STATE_OFFHOOK) {
473            resumeCall();
474        } else {
475            launchEmergencyDialer();
476        }
477    }
478
479    private void resumeCall() {
480        ITelephony phone = ITelephony.Stub.asInterface(ServiceManager.checkService("phone"));
481        if (phone != null) {
482            try {
483                phone.showCallScreen();
484            } catch (RemoteException e) {
485                Log.e(TAG, "Error calling ITelephony service: " + e);
486            }
487        }
488    }
489
490    private void launchEmergencyDialer() {
491        Intent intent = new Intent(ACTION_EMERGENCY_DIAL);
492        intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK
493                        | Intent.FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS);
494        startActivity(intent);
495    }
496}
497