19066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/*
29066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Copyright (C) 2008 The Android Open Source Project
39066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
49066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Licensed under the Apache License, Version 2.0 (the "License");
59066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * you may not use this file except in compliance with the License.
69066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * You may obtain a copy of the License at
79066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
89066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *      http://www.apache.org/licenses/LICENSE-2.0
99066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Unless required by applicable law or agreed to in writing, software
119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * distributed under the License is distributed on an "AS IS" BASIS,
129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * See the License for the specific language governing permissions and
149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * limitations under the License.
159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */
169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectpackage com.android.server;
189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport com.android.server.am.ActivityManagerService;
204f8ecd80296508a1dc69d3f3a23fd91e962c2784Jeff Brownimport com.android.server.power.PowerManagerService;
219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.app.AlarmManager;
239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.app.PendingIntent;
249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.content.BroadcastReceiver;
259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.content.ContentResolver;
269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.content.Context;
279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.content.Intent;
289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.content.IntentFilter;
29a4d8204e3068b9d8d6908d4cf3440e81967867a3Jeff Brownimport android.os.BatteryManager;
309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.os.Debug;
319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.os.Handler;
329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.os.Message;
339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.os.Process;
346ffce2e9a3c57634bb73f8ff133ca680f8070d5dSuchi Amalapurapuimport android.os.ServiceManager;
359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.os.SystemClock;
369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.os.SystemProperties;
379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.util.EventLog;
389bdc94b7a42a07d7dafcdf2cbadbb9c736b979d2Dan Egnorimport android.util.Log;
398a9b22056b13477f59df934928c00c58b5871c95Joe Onoratoimport android.util.Slog;
409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
419bdc94b7a42a07d7dafcdf2cbadbb9c736b979d2Dan Egnorimport java.io.File;
425df1d871feabee23b16a69ee48695fd892017517Colin Crossimport java.io.FileWriter;
435df1d871feabee23b16a69ee48695fd892017517Colin Crossimport java.io.IOException;
449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.util.ArrayList;
459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.util.Calendar;
469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/** This class calls its monitor every minute. Killing this process if they don't return **/
489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectpublic class Watchdog extends Thread {
499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static final String TAG = "Watchdog";
5043a17654cf4bfe7f1ec22bd8b7b32daccdf27c09Joe Onorato    static final boolean localLOGV = false || false;
519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // Set this to true to use debug default values.
539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static final boolean DB = false;
549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
55ecaa7b41ca49154ceaa9a7504eb0a86b89a96026Christopher Tate    // Set this to true to have the watchdog record kernel thread stacks when it fires
56ecaa7b41ca49154ceaa9a7504eb0a86b89a96026Christopher Tate    static final boolean RECORD_KERNEL_THREADS = true;
57ecaa7b41ca49154ceaa9a7504eb0a86b89a96026Christopher Tate
589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static final int MONITOR = 2718;
599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
60cf2317ef13e35cf1bcd5ba27be686c7f2609ac38Mathias Agopian    static final int TIME_TO_RESTART = DB ? 15*1000 : 60*1000;
616ee412d51d8b601580cfb4b7be4f676b7ec76afdChristopher Tate    static final int TIME_TO_WAIT = TIME_TO_RESTART / 2;
629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static final int MEMCHECK_DEFAULT_MIN_SCREEN_OFF = DB ? 1*60 : 5*60;   // 5 minutes
649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static final int MEMCHECK_DEFAULT_MIN_ALARM = DB ? 1*60 : 3*60;        // 3 minutes
659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static final int MEMCHECK_DEFAULT_RECHECK_INTERVAL = DB ? 1*60 : 5*60; // 5 minutes
669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static final int REBOOT_DEFAULT_INTERVAL = DB ? 1 : 0;                 // never force reboot
689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static final int REBOOT_DEFAULT_START_TIME = 3*60*60;                  // 3:00am
699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static final int REBOOT_DEFAULT_WINDOW = 60*60;                        // within 1 hour
709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static final String REBOOT_ACTION = "com.android.service.Watchdog.REBOOT";
729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
73f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn    static final String[] NATIVE_STACKS_OF_INTEREST = new String[] {
74f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn        "/system/bin/mediaserver",
75f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn        "/system/bin/sdcard",
76f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn        "/system/bin/surfaceflinger"
77f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn    };
78f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn
799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static Watchdog sWatchdog;
809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /* This handler will be used to post message back onto the main thread */
829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    final Handler mHandler;
839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    final ArrayList<Monitor> mMonitors = new ArrayList<Monitor>();
849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    ContentResolver mResolver;
859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    BatteryService mBattery;
869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    PowerManagerService mPower;
879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    AlarmManagerService mAlarm;
889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    ActivityManagerService mActivity;
899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    boolean mCompleted;
909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    boolean mForceKillSystem;
919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Monitor mCurrentMonitor;
929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    int mPhonePid;
949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    final Calendar mCalendar = Calendar.getInstance();
969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    int mMinScreenOff = MEMCHECK_DEFAULT_MIN_SCREEN_OFF;
979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    int mMinAlarm = MEMCHECK_DEFAULT_MIN_ALARM;
989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    boolean mNeedScheduledCheck;
999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    PendingIntent mCheckupIntent;
1009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    PendingIntent mRebootIntent;
1019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    long mBootTime;
1039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    int mRebootInterval;
1049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    boolean mReqRebootNoWait;     // should wait for one interval before reboot?
1069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    int mReqRebootInterval = -1;  // >= 0 if a reboot has been requested
1079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    int mReqRebootStartTime = -1; // >= 0 if a specific start time has been requested
1089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    int mReqRebootWindow = -1;    // >= 0 if a specific window has been requested
1099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    int mReqMinScreenOff = -1;    // >= 0 if a specific screen off time has been requested
1109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    int mReqMinNextAlarm = -1;    // >= 0 if specific time to next alarm has been requested
1119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    int mReqRecheckInterval= -1;  // >= 0 if a specific recheck interval has been requested
1129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
1149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Used for scheduling monitor callbacks and checking memory usage.
1159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
1169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    final class HeartbeatHandler extends Handler {
1179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        @Override
1189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public void handleMessage(Message msg) {
1199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            switch (msg.what) {
1209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                case MONITOR: {
1219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // See if we should force a reboot.
1229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    int rebootInterval = mReqRebootInterval >= 0
1234de9936e85696208dfe91d1c40e3e5226e57634aJeff Sharkey                            ? mReqRebootInterval : REBOOT_DEFAULT_INTERVAL;
1249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    if (mRebootInterval != rebootInterval) {
1259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        mRebootInterval = rebootInterval;
1269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        // We have been running long enough that a reboot can
1279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        // be considered...
1289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        checkReboot(false);
1299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    }
1309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    final int size = mMonitors.size();
1329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    for (int i = 0 ; i < size ; i++) {
1339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        mCurrentMonitor = mMonitors.get(i);
1349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        mCurrentMonitor.monitor();
1359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    }
1369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    synchronized (Watchdog.this) {
1389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        mCompleted = true;
1399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        mCurrentMonitor = null;
1409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    }
1419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                } break;
1429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
1439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
1449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    final class RebootReceiver extends BroadcastReceiver {
1479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        @Override
1489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public void onReceive(Context c, Intent intent) {
1498a9b22056b13477f59df934928c00c58b5871c95Joe Onorato            if (localLOGV) Slog.v(TAG, "Alarm went off, checking reboot.");
1509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            checkReboot(true);
1519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
1529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    final class RebootRequestReceiver extends BroadcastReceiver {
1559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        @Override
1569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public void onReceive(Context c, Intent intent) {
1579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mReqRebootNoWait = intent.getIntExtra("nowait", 0) != 0;
1589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mReqRebootInterval = intent.getIntExtra("interval", -1);
1599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mReqRebootStartTime = intent.getIntExtra("startTime", -1);
1609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mReqRebootWindow = intent.getIntExtra("window", -1);
1619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mReqMinScreenOff = intent.getIntExtra("minScreenOff", -1);
1629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mReqMinNextAlarm = intent.getIntExtra("minNextAlarm", -1);
1639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mReqRecheckInterval = intent.getIntExtra("recheckInterval", -1);
164ab5c49c7e7f5f61040650109a76f38a443fb852dDoug Zongker            EventLog.writeEvent(EventLogTags.WATCHDOG_REQUESTED_REBOOT,
1659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    mReqRebootNoWait ? 1 : 0, mReqRebootInterval,
1669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                            mReqRecheckInterval, mReqRebootStartTime,
1679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    mReqRebootWindow, mReqMinScreenOff, mReqMinNextAlarm);
1689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            checkReboot(true);
1699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
1709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public interface Monitor {
1739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        void monitor();
1749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static Watchdog getInstance() {
1779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (sWatchdog == null) {
1789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            sWatchdog = new Watchdog();
1799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
1809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return sWatchdog;
1829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private Watchdog() {
1859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        super("watchdog");
1869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mHandler = new HeartbeatHandler();
1879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void init(Context context, BatteryService battery,
1909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            PowerManagerService power, AlarmManagerService alarm,
1919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            ActivityManagerService activity) {
1929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mResolver = context.getContentResolver();
1939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mBattery = battery;
1949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mPower = power;
1959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mAlarm = alarm;
1969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mActivity = activity;
1979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        context.registerReceiver(new RebootReceiver(),
1999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                new IntentFilter(REBOOT_ACTION));
2009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mRebootIntent = PendingIntent.getBroadcast(context,
2019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                0, new Intent(REBOOT_ACTION), 0);
2029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        context.registerReceiver(new RebootRequestReceiver(),
2049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                new IntentFilter(Intent.ACTION_REBOOT),
2059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                android.Manifest.permission.REBOOT, null);
2069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mBootTime = System.currentTimeMillis();
2089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
210c27181c7f3e11170ec82807cfa416f0a906ff574Christopher Tate    public void processStarted(String name, int pid) {
2119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        synchronized (this) {
2129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if ("com.android.phone".equals(name)) {
2139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mPhonePid = pid;
2149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
2159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
2169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void addMonitor(Monitor monitor) {
2199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        synchronized (this) {
2209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (isAlive()) {
2219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                throw new RuntimeException("Monitors can't be added while the Watchdog is running");
2229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
2239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mMonitors.add(monitor);
2249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
2259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    void checkReboot(boolean fromAlarm) {
2289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int rebootInterval = mReqRebootInterval >= 0 ? mReqRebootInterval
2294de9936e85696208dfe91d1c40e3e5226e57634aJeff Sharkey                : REBOOT_DEFAULT_INTERVAL;
2309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mRebootInterval = rebootInterval;
2319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (rebootInterval <= 0) {
2329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // No reboot interval requested.
2338a9b22056b13477f59df934928c00c58b5871c95Joe Onorato            if (localLOGV) Slog.v(TAG, "No need to schedule a reboot alarm!");
2349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mAlarm.remove(mRebootIntent);
2359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return;
2369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
2379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        long rebootStartTime = mReqRebootStartTime >= 0 ? mReqRebootStartTime
2394de9936e85696208dfe91d1c40e3e5226e57634aJeff Sharkey                : REBOOT_DEFAULT_START_TIME;
2409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        long rebootWindowMillis = (mReqRebootWindow >= 0 ? mReqRebootWindow
2414de9936e85696208dfe91d1c40e3e5226e57634aJeff Sharkey                : REBOOT_DEFAULT_WINDOW) * 1000;
2429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        long recheckInterval = (mReqRecheckInterval >= 0 ? mReqRecheckInterval
2434de9936e85696208dfe91d1c40e3e5226e57634aJeff Sharkey                : MEMCHECK_DEFAULT_RECHECK_INTERVAL) * 1000;
2449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        retrieveBrutalityAmount();
2469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        long realStartTime;
2489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        long now;
2499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        synchronized (this) {
2519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            now = System.currentTimeMillis();
2529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            realStartTime = computeCalendarTime(mCalendar, now,
2539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    rebootStartTime);
2549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            long rebootIntervalMillis = rebootInterval*24*60*60*1000;
2569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (DB || mReqRebootNoWait ||
2579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    (now-mBootTime) >= (rebootIntervalMillis-rebootWindowMillis)) {
2589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (fromAlarm && rebootWindowMillis <= 0) {
2599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // No reboot window -- just immediately reboot.
260ab5c49c7e7f5f61040650109a76f38a443fb852dDoug Zongker                    EventLog.writeEvent(EventLogTags.WATCHDOG_SCHEDULED_REBOOT, now,
2619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                            (int)rebootIntervalMillis, (int)rebootStartTime*1000,
2629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                            (int)rebootWindowMillis, "");
2639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    rebootSystem("Checkin scheduled forced");
2649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    return;
2659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
2669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // Are we within the reboot window?
2689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (now < realStartTime) {
2699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // Schedule alarm for next check interval.
2709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    realStartTime = computeCalendarTime(mCalendar,
2719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                            now, rebootStartTime);
2729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                } else if (now < (realStartTime+rebootWindowMillis)) {
2739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    String doit = shouldWeBeBrutalLocked(now);
274ab5c49c7e7f5f61040650109a76f38a443fb852dDoug Zongker                    EventLog.writeEvent(EventLogTags.WATCHDOG_SCHEDULED_REBOOT, now,
2759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                            (int)rebootInterval, (int)rebootStartTime*1000,
2769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                            (int)rebootWindowMillis, doit != null ? doit : "");
2779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    if (doit == null) {
2789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        rebootSystem("Checked scheduled range");
2799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        return;
2809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    }
2819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // Schedule next alarm either within the window or in the
2839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // next interval.
2849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    if ((now+recheckInterval) >= (realStartTime+rebootWindowMillis)) {
2859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        realStartTime = computeCalendarTime(mCalendar,
2869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                now + rebootIntervalMillis, rebootStartTime);
2879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    } else {
2889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        realStartTime = now + recheckInterval;
2899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    }
2909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                } else {
2919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // Schedule alarm for next check interval.
2929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    realStartTime = computeCalendarTime(mCalendar,
2939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                            now + rebootIntervalMillis, rebootStartTime);
2949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
2959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
2969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
2979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2988a9b22056b13477f59df934928c00c58b5871c95Joe Onorato        if (localLOGV) Slog.v(TAG, "Scheduling next reboot alarm for "
2999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                + ((realStartTime-now)/1000/60) + "m from now");
3009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mAlarm.remove(mRebootIntent);
3019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mAlarm.set(AlarmManager.RTC_WAKEUP, realStartTime, mRebootIntent);
3029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
3059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Perform a full reboot of the system.
3069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
3079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    void rebootSystem(String reason) {
3088a9b22056b13477f59df934928c00c58b5871c95Joe Onorato        Slog.i(TAG, "Rebooting system because: " + reason);
3096ffce2e9a3c57634bb73f8ff133ca680f8070d5dSuchi Amalapurapu        PowerManagerService pms = (PowerManagerService) ServiceManager.getService("power");
310c428aae6429c3fd5e2037c3793af399d9f6e23bfDianne Hackborn        pms.reboot(false, reason, false);
3119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
3149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Load the current Gservices settings for when
3159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * {@link #shouldWeBeBrutalLocked} will allow the brutality to happen.
3169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Must not be called with the lock held.
3179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
3189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    void retrieveBrutalityAmount() {
3199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mMinScreenOff = (mReqMinScreenOff >= 0 ? mReqMinScreenOff
3204de9936e85696208dfe91d1c40e3e5226e57634aJeff Sharkey                : MEMCHECK_DEFAULT_MIN_SCREEN_OFF) * 1000;
3219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mMinAlarm = (mReqMinNextAlarm >= 0 ? mReqMinNextAlarm
3224de9936e85696208dfe91d1c40e3e5226e57634aJeff Sharkey                : MEMCHECK_DEFAULT_MIN_ALARM) * 1000;
3239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
3269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Determine whether it is a good time to kill, crash, or otherwise
3279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * plunder the current situation for the overall long-term benefit of
3289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * the world.
3299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
3309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param curTime The current system time.
3319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return Returns null if this is a good time, else a String with the
3329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * text of why it is not a good time.
3339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
3349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    String shouldWeBeBrutalLocked(long curTime) {
335a4d8204e3068b9d8d6908d4cf3440e81967867a3Jeff Brown        if (mBattery == null || !mBattery.isPowered(BatteryManager.BATTERY_PLUGGED_ANY)) {
3369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return "battery";
3379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mMinScreenOff >= 0 && (mPower == null ||
3409630704ed3b265f008a8f64ec60a33cf9dcd3345Jeff Brown                mPower.timeSinceScreenWasLastOn() < mMinScreenOff)) {
3419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return "screen";
3429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mMinAlarm >= 0 && (mAlarm == null ||
3459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mAlarm.timeToNextAlarm() < mMinAlarm)) {
3469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return "alarm";
3479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return null;
3509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static long computeCalendarTime(Calendar c, long curTime,
3539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            long secondsSinceMidnight) {
3549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // start with now
3569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        c.setTimeInMillis(curTime);
3579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int val = (int)secondsSinceMidnight / (60*60);
3599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        c.set(Calendar.HOUR_OF_DAY, val);
3609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        secondsSinceMidnight -= val * (60*60);
3619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        val = (int)secondsSinceMidnight / 60;
3629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        c.set(Calendar.MINUTE, val);
3639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        c.set(Calendar.SECOND, (int)secondsSinceMidnight - (val*60));
3649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        c.set(Calendar.MILLISECOND, 0);
3659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        long newTime = c.getTimeInMillis();
3679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (newTime < curTime) {
3689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // The given time (in seconds since midnight) has already passed for today, so advance
3699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // by one day (due to daylight savings, etc., the delta may differ from 24 hours).
3709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            c.add(Calendar.DAY_OF_MONTH, 1);
3719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            newTime = c.getTimeInMillis();
3729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return newTime;
3759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
3789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void run() {
3796ee412d51d8b601580cfb4b7be4f676b7ec76afdChristopher Tate        boolean waitedHalf = false;
3809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        while (true) {
3819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mCompleted = false;
3829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mHandler.sendEmptyMessage(MONITOR);
3839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            synchronized (this) {
3859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                long timeout = TIME_TO_WAIT;
3869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // NOTE: We use uptimeMillis() here because we do not want to increment the time we
3889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // wait while asleep. If the device is asleep then the thing that we are waiting
3896ee412d51d8b601580cfb4b7be4f676b7ec76afdChristopher Tate                // to timeout on is asleep as well and won't have a chance to run, causing a false
3909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // positive on when to kill things.
3919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                long start = SystemClock.uptimeMillis();
3929bdc94b7a42a07d7dafcdf2cbadbb9c736b979d2Dan Egnor                while (timeout > 0 && !mForceKillSystem) {
3939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    try {
3949bdc94b7a42a07d7dafcdf2cbadbb9c736b979d2Dan Egnor                        wait(timeout);  // notifyAll() is called when mForceKillSystem is set
3959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    } catch (InterruptedException e) {
3969bdc94b7a42a07d7dafcdf2cbadbb9c736b979d2Dan Egnor                        Log.wtf(TAG, e);
3979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    }
3989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    timeout = TIME_TO_WAIT - (SystemClock.uptimeMillis() - start);
3999bdc94b7a42a07d7dafcdf2cbadbb9c736b979d2Dan Egnor                }
4009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (mCompleted && !mForceKillSystem) {
4029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // The monitors have returned.
4036ee412d51d8b601580cfb4b7be4f676b7ec76afdChristopher Tate                    waitedHalf = false;
4046ee412d51d8b601580cfb4b7be4f676b7ec76afdChristopher Tate                    continue;
4056ee412d51d8b601580cfb4b7be4f676b7ec76afdChristopher Tate                }
4066ee412d51d8b601580cfb4b7be4f676b7ec76afdChristopher Tate
4076ee412d51d8b601580cfb4b7be4f676b7ec76afdChristopher Tate                if (!waitedHalf) {
4086ee412d51d8b601580cfb4b7be4f676b7ec76afdChristopher Tate                    // We've waited half the deadlock-detection interval.  Pull a stack
4096ee412d51d8b601580cfb4b7be4f676b7ec76afdChristopher Tate                    // trace and wait another half.
4106b1afebdaca6c27d49a243c4283e5e2e4924de8cDianne Hackborn                    ArrayList<Integer> pids = new ArrayList<Integer>();
4116ee412d51d8b601580cfb4b7be4f676b7ec76afdChristopher Tate                    pids.add(Process.myPid());
412f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn                    ActivityManagerService.dumpStackTraces(true, pids, null, null,
413f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn                            NATIVE_STACKS_OF_INTEREST);
4146ee412d51d8b601580cfb4b7be4f676b7ec76afdChristopher Tate                    waitedHalf = true;
4159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    continue;
4169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
4179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
4189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // If we got here, that means that the system is most likely hung.
420784827b27cf4cd82bf074b571e63cb5e660c54afJean-Baptiste Queru            // First collect stack traces from all threads of the system process.
421784827b27cf4cd82bf074b571e63cb5e660c54afJean-Baptiste Queru            // Then kill this process so that the system will restart.
4229bdc94b7a42a07d7dafcdf2cbadbb9c736b979d2Dan Egnor
4239765c72eeaba9ed0f704dd992e7a954f38f20abdBrad Fitzpatrick            final String name = (mCurrentMonitor != null) ?
4246b1afebdaca6c27d49a243c4283e5e2e4924de8cDianne Hackborn                    mCurrentMonitor.getClass().getName() : "null";
425ab5c49c7e7f5f61040650109a76f38a443fb852dDoug Zongker            EventLog.writeEvent(EventLogTags.WATCHDOG, name);
4269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4276b1afebdaca6c27d49a243c4283e5e2e4924de8cDianne Hackborn            ArrayList<Integer> pids = new ArrayList<Integer>();
4289bdc94b7a42a07d7dafcdf2cbadbb9c736b979d2Dan Egnor            pids.add(Process.myPid());
4294bded0744a07152c1e7ae4cb8110c74ec89a67b1Dan Egnor            if (mPhonePid > 0) pids.add(mPhonePid);
4306ee412d51d8b601580cfb4b7be4f676b7ec76afdChristopher Tate            // Pass !waitedHalf so that just in case we somehow wind up here without having
4316ee412d51d8b601580cfb4b7be4f676b7ec76afdChristopher Tate            // dumped the halfway stacks, we properly re-initialize the trace file.
4329765c72eeaba9ed0f704dd992e7a954f38f20abdBrad Fitzpatrick            final File stack = ActivityManagerService.dumpStackTraces(
433f72467ad9843bf5d4b75fb308386e77ebb5c3447Dianne Hackborn                    !waitedHalf, pids, null, null, NATIVE_STACKS_OF_INTEREST);
4344bded0744a07152c1e7ae4cb8110c74ec89a67b1Dan Egnor
4354bded0744a07152c1e7ae4cb8110c74ec89a67b1Dan Egnor            // Give some extra time to make sure the stack traces get written.
4364bded0744a07152c1e7ae4cb8110c74ec89a67b1Dan Egnor            // The system's been hanging for a minute, another second or two won't hurt much.
4374bded0744a07152c1e7ae4cb8110c74ec89a67b1Dan Egnor            SystemClock.sleep(2000);
4384bded0744a07152c1e7ae4cb8110c74ec89a67b1Dan Egnor
439ecaa7b41ca49154ceaa9a7504eb0a86b89a96026Christopher Tate            // Pull our own kernel thread stacks as well if we're configured for that
440ecaa7b41ca49154ceaa9a7504eb0a86b89a96026Christopher Tate            if (RECORD_KERNEL_THREADS) {
441ecaa7b41ca49154ceaa9a7504eb0a86b89a96026Christopher Tate                dumpKernelStackTraces();
442ecaa7b41ca49154ceaa9a7504eb0a86b89a96026Christopher Tate            }
443ecaa7b41ca49154ceaa9a7504eb0a86b89a96026Christopher Tate
4445df1d871feabee23b16a69ee48695fd892017517Colin Cross            // Trigger the kernel to dump all blocked threads to the kernel log
4455df1d871feabee23b16a69ee48695fd892017517Colin Cross            try {
4465df1d871feabee23b16a69ee48695fd892017517Colin Cross                FileWriter sysrq_trigger = new FileWriter("/proc/sysrq-trigger");
4475df1d871feabee23b16a69ee48695fd892017517Colin Cross                sysrq_trigger.write("w");
4485df1d871feabee23b16a69ee48695fd892017517Colin Cross                sysrq_trigger.close();
4495df1d871feabee23b16a69ee48695fd892017517Colin Cross            } catch (IOException e) {
4505df1d871feabee23b16a69ee48695fd892017517Colin Cross                Slog.e(TAG, "Failed to write to /proc/sysrq-trigger");
4515df1d871feabee23b16a69ee48695fd892017517Colin Cross                Slog.e(TAG, e.getMessage());
4525df1d871feabee23b16a69ee48695fd892017517Colin Cross            }
4535df1d871feabee23b16a69ee48695fd892017517Colin Cross
4549765c72eeaba9ed0f704dd992e7a954f38f20abdBrad Fitzpatrick            // Try to add the error to the dropbox, but assuming that the ActivityManager
4559765c72eeaba9ed0f704dd992e7a954f38f20abdBrad Fitzpatrick            // itself may be deadlocked.  (which has happened, causing this statement to
4569765c72eeaba9ed0f704dd992e7a954f38f20abdBrad Fitzpatrick            // deadlock and the watchdog as a whole to be ineffective)
4579765c72eeaba9ed0f704dd992e7a954f38f20abdBrad Fitzpatrick            Thread dropboxThread = new Thread("watchdogWriteToDropbox") {
4589765c72eeaba9ed0f704dd992e7a954f38f20abdBrad Fitzpatrick                    public void run() {
4599765c72eeaba9ed0f704dd992e7a954f38f20abdBrad Fitzpatrick                        mActivity.addErrorToDropBox(
460a353d2654a98b292469d2559cb4424b13d779924Jeff Sharkey                                "watchdog", null, "system_server", null, null,
461a353d2654a98b292469d2559cb4424b13d779924Jeff Sharkey                                name, null, stack, null);
4629765c72eeaba9ed0f704dd992e7a954f38f20abdBrad Fitzpatrick                    }
4639765c72eeaba9ed0f704dd992e7a954f38f20abdBrad Fitzpatrick                };
4649765c72eeaba9ed0f704dd992e7a954f38f20abdBrad Fitzpatrick            dropboxThread.start();
4659765c72eeaba9ed0f704dd992e7a954f38f20abdBrad Fitzpatrick            try {
4669765c72eeaba9ed0f704dd992e7a954f38f20abdBrad Fitzpatrick                dropboxThread.join(2000);  // wait up to 2 seconds for it to return.
4679765c72eeaba9ed0f704dd992e7a954f38f20abdBrad Fitzpatrick            } catch (InterruptedException ignored) {}
4689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
469784827b27cf4cd82bf074b571e63cb5e660c54afJean-Baptiste Queru            // Only kill the process if the debugger is not attached.
4709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (!Debug.isDebuggerConnected()) {
4719bdc94b7a42a07d7dafcdf2cbadbb9c736b979d2Dan Egnor                Slog.w(TAG, "*** WATCHDOG KILLING SYSTEM PROCESS: " + name);
472784827b27cf4cd82bf074b571e63cb5e660c54afJean-Baptiste Queru                Process.killProcess(Process.myPid());
473784827b27cf4cd82bf074b571e63cb5e660c54afJean-Baptiste Queru                System.exit(10);
4749bdc94b7a42a07d7dafcdf2cbadbb9c736b979d2Dan Egnor            } else {
4759bdc94b7a42a07d7dafcdf2cbadbb9c736b979d2Dan Egnor                Slog.w(TAG, "Debugger connected: Watchdog is *not* killing the system process");
4769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
4776ee412d51d8b601580cfb4b7be4f676b7ec76afdChristopher Tate
4786ee412d51d8b601580cfb4b7be4f676b7ec76afdChristopher Tate            waitedHalf = false;
4799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
481ecaa7b41ca49154ceaa9a7504eb0a86b89a96026Christopher Tate
482ecaa7b41ca49154ceaa9a7504eb0a86b89a96026Christopher Tate    private File dumpKernelStackTraces() {
483ecaa7b41ca49154ceaa9a7504eb0a86b89a96026Christopher Tate        String tracesPath = SystemProperties.get("dalvik.vm.stack-trace-file", null);
484ecaa7b41ca49154ceaa9a7504eb0a86b89a96026Christopher Tate        if (tracesPath == null || tracesPath.length() == 0) {
485ecaa7b41ca49154ceaa9a7504eb0a86b89a96026Christopher Tate            return null;
486ecaa7b41ca49154ceaa9a7504eb0a86b89a96026Christopher Tate        }
487ecaa7b41ca49154ceaa9a7504eb0a86b89a96026Christopher Tate
488ecaa7b41ca49154ceaa9a7504eb0a86b89a96026Christopher Tate        native_dumpKernelStacks(tracesPath);
489ecaa7b41ca49154ceaa9a7504eb0a86b89a96026Christopher Tate        return new File(tracesPath);
490ecaa7b41ca49154ceaa9a7504eb0a86b89a96026Christopher Tate    }
491ecaa7b41ca49154ceaa9a7504eb0a86b89a96026Christopher Tate
492ecaa7b41ca49154ceaa9a7504eb0a86b89a96026Christopher Tate    private native void native_dumpKernelStacks(String tracesPath);
4939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
494