1 /* 2 * Copyright (C) 2009 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.content.BroadcastReceiver; 20 import android.content.ContentResolver; 21 import android.content.Context; 22 import android.content.Intent; 23 import android.content.SharedPreferences; 24 import android.net.Downloads; 25 import android.os.Build; 26 import android.os.DropBoxManager; 27 import android.os.FileObserver; 28 import android.os.FileUtils; 29 import android.os.RecoverySystem; 30 import android.os.SystemProperties; 31 import android.provider.Settings; 32 import android.util.Slog; 33 34 import java.io.File; 35 import java.io.IOException; 36 37 /** 38 * Performs a number of miscellaneous, non-system-critical actions 39 * after the system has finished booting. 40 */ 41 public class BootReceiver extends BroadcastReceiver { 42 private static final String TAG = "BootReceiver"; 43 44 // Maximum size of a logged event (files get truncated if they're longer) 45 private static final int LOG_SIZE = 65536; 46 47 private static final File TOMBSTONE_DIR = new File("/data/tombstones"); 48 49 // The pre-froyo package and class of the system updater, which 50 // ran in the system process. We need to remove its packages here 51 // in order to clean up after a pre-froyo-to-froyo update. 52 private static final String OLD_UPDATER_PACKAGE = 53 "com.google.android.systemupdater"; 54 private static final String OLD_UPDATER_CLASS = 55 "com.google.android.systemupdater.SystemUpdateReceiver"; 56 57 // Keep a reference to the observer so the finalizer doesn't disable it. 58 private static FileObserver sTombstoneObserver = null; 59 60 @Override onReceive(final Context context, Intent intent)61 public void onReceive(final Context context, Intent intent) { 62 try { 63 // Start the load average overlay, if activated 64 ContentResolver res = context.getContentResolver(); 65 if (Settings.System.getInt(res, Settings.System.SHOW_PROCESSES, 0) != 0) { 66 Intent loadavg = new Intent(context, com.android.server.LoadAverageService.class); 67 context.startService(loadavg); 68 } 69 } catch (Exception e) { 70 Slog.e(TAG, "Can't start load average service", e); 71 } 72 73 // Log boot events in the background to avoid blocking the main thread with I/O 74 new Thread() { 75 @Override 76 public void run() { 77 try { 78 logBootEvents(context); 79 } catch (Exception e) { 80 Slog.e(TAG, "Can't log boot events", e); 81 } 82 try { 83 removeOldUpdatePackages(context); 84 } catch (Exception e) { 85 Slog.e(TAG, "Can't remove old update packages", e); 86 } 87 88 } 89 }.start(); 90 } 91 removeOldUpdatePackages(Context ctx)92 private void removeOldUpdatePackages(Context ctx) { 93 Downloads.ByUri.removeAllDownloadsByPackage( 94 ctx, OLD_UPDATER_PACKAGE, OLD_UPDATER_CLASS); 95 } 96 logBootEvents(Context ctx)97 private void logBootEvents(Context ctx) throws IOException { 98 final DropBoxManager db = (DropBoxManager) ctx.getSystemService(Context.DROPBOX_SERVICE); 99 final SharedPreferences prefs = ctx.getSharedPreferences("log_files", Context.MODE_PRIVATE); 100 final String headers = new StringBuilder(512) 101 .append("Build: ").append(Build.FINGERPRINT).append("\n") 102 .append("Hardware: ").append(Build.BOARD).append("\n") 103 .append("Bootloader: ").append(Build.BOOTLOADER).append("\n") 104 .append("Radio: ").append(Build.RADIO).append("\n") 105 .append("Kernel: ") 106 .append(FileUtils.readTextFile(new File("/proc/version"), 1024, "...\n")) 107 .append("\n").toString(); 108 109 String recovery = RecoverySystem.handleAftermath(); 110 if (recovery != null && db != null) { 111 db.addText("SYSTEM_RECOVERY_LOG", headers + recovery); 112 } 113 114 if (SystemProperties.getLong("ro.runtime.firstboot", 0) == 0) { 115 String now = Long.toString(System.currentTimeMillis()); 116 SystemProperties.set("ro.runtime.firstboot", now); 117 if (db != null) db.addText("SYSTEM_BOOT", headers); 118 119 // Negative sizes mean to take the *tail* of the file (see FileUtils.readTextFile()) 120 addFileToDropBox(db, prefs, headers, "/proc/last_kmsg", 121 -LOG_SIZE, "SYSTEM_LAST_KMSG"); 122 addFileToDropBox(db, prefs, headers, "/cache/recovery/log", 123 -LOG_SIZE, "SYSTEM_RECOVERY_LOG"); 124 addFileToDropBox(db, prefs, headers, "/data/dontpanic/apanic_console", 125 -LOG_SIZE, "APANIC_CONSOLE"); 126 addFileToDropBox(db, prefs, headers, "/data/dontpanic/apanic_threads", 127 -LOG_SIZE, "APANIC_THREADS"); 128 } else { 129 if (db != null) db.addText("SYSTEM_RESTART", headers); 130 } 131 132 // Scan existing tombstones (in case any new ones appeared) 133 File[] tombstoneFiles = TOMBSTONE_DIR.listFiles(); 134 for (int i = 0; tombstoneFiles != null && i < tombstoneFiles.length; i++) { 135 addFileToDropBox(db, prefs, headers, tombstoneFiles[i].getPath(), 136 LOG_SIZE, "SYSTEM_TOMBSTONE"); 137 } 138 139 // Start watching for new tombstone files; will record them as they occur. 140 // This gets registered with the singleton file observer thread. 141 sTombstoneObserver = new FileObserver(TOMBSTONE_DIR.getPath(), FileObserver.CLOSE_WRITE) { 142 @Override 143 public void onEvent(int event, String path) { 144 try { 145 String filename = new File(TOMBSTONE_DIR, path).getPath(); 146 addFileToDropBox(db, prefs, headers, filename, LOG_SIZE, "SYSTEM_TOMBSTONE"); 147 } catch (IOException e) { 148 Slog.e(TAG, "Can't log tombstone", e); 149 } 150 } 151 }; 152 153 sTombstoneObserver.startWatching(); 154 } 155 addFileToDropBox( DropBoxManager db, SharedPreferences prefs, String headers, String filename, int maxSize, String tag)156 private static void addFileToDropBox( 157 DropBoxManager db, SharedPreferences prefs, 158 String headers, String filename, int maxSize, String tag) throws IOException { 159 if (db == null || !db.isTagEnabled(tag)) return; // Logging disabled 160 161 File file = new File(filename); 162 long fileTime = file.lastModified(); 163 if (fileTime <= 0) return; // File does not exist 164 165 if (prefs != null) { 166 long lastTime = prefs.getLong(filename, 0); 167 if (lastTime == fileTime) return; // Already logged this particular file 168 // TODO: move all these SharedPreferences Editor commits 169 // outside this function to the end of logBootEvents 170 prefs.edit().putLong(filename, fileTime).apply(); 171 } 172 173 Slog.i(TAG, "Copying " + filename + " to DropBox (" + tag + ")"); 174 db.addText(tag, headers + FileUtils.readTextFile(file, maxSize, "[[TRUNCATED]]\n")); 175 } 176 } 177