Merge "Framework: Remove SamplingProfiler"
This commit is contained in:
@@ -116,7 +116,6 @@ import com.android.internal.app.IVoiceInteractor;
|
||||
import com.android.internal.content.ReferrerIntent;
|
||||
import com.android.internal.os.BinderInternal;
|
||||
import com.android.internal.os.RuntimeInit;
|
||||
import com.android.internal.os.SamplingProfilerIntegration;
|
||||
import com.android.internal.os.SomeArgs;
|
||||
import com.android.internal.util.ArrayUtils;
|
||||
import com.android.internal.util.FastPrintWriter;
|
||||
@@ -1500,7 +1499,6 @@ public final class ActivityThread {
|
||||
handlePauseActivity((IBinder) args.arg1, false,
|
||||
(args.argi1 & USER_LEAVING) != 0, args.argi2,
|
||||
(args.argi1 & DONT_REPORT) != 0, args.argi3);
|
||||
maybeSnapshot();
|
||||
Trace.traceEnd(Trace.TRACE_TAG_ACTIVITY_MANAGER);
|
||||
} break;
|
||||
case PAUSE_ACTIVITY_FINISHING: {
|
||||
@@ -1570,7 +1568,6 @@ public final class ActivityThread {
|
||||
case RECEIVER:
|
||||
Trace.traceBegin(Trace.TRACE_TAG_ACTIVITY_MANAGER, "broadcastReceiveComp");
|
||||
handleReceiver((ReceiverData)msg.obj);
|
||||
maybeSnapshot();
|
||||
Trace.traceEnd(Trace.TRACE_TAG_ACTIVITY_MANAGER);
|
||||
break;
|
||||
case CREATE_SERVICE:
|
||||
@@ -1596,7 +1593,6 @@ public final class ActivityThread {
|
||||
case STOP_SERVICE:
|
||||
Trace.traceBegin(Trace.TRACE_TAG_ACTIVITY_MANAGER, "serviceStop");
|
||||
handleStopService((IBinder)msg.obj);
|
||||
maybeSnapshot();
|
||||
Trace.traceEnd(Trace.TRACE_TAG_ACTIVITY_MANAGER);
|
||||
break;
|
||||
case CONFIGURATION_CHANGED:
|
||||
@@ -1737,32 +1733,6 @@ public final class ActivityThread {
|
||||
}
|
||||
if (DEBUG_MESSAGES) Slog.v(TAG, "<<< done: " + codeToString(msg.what));
|
||||
}
|
||||
|
||||
private void maybeSnapshot() {
|
||||
if (mBoundApplication != null && SamplingProfilerIntegration.isEnabled()) {
|
||||
// convert the *private* ActivityThread.PackageInfo to *public* known
|
||||
// android.content.pm.PackageInfo
|
||||
String packageName = mBoundApplication.info.mPackageName;
|
||||
android.content.pm.PackageInfo packageInfo = null;
|
||||
try {
|
||||
Context context = getSystemContext();
|
||||
if(context == null) {
|
||||
Log.e(TAG, "cannot get a valid context");
|
||||
return;
|
||||
}
|
||||
PackageManager pm = context.getPackageManager();
|
||||
if(pm == null) {
|
||||
Log.e(TAG, "cannot get a valid PackageManager");
|
||||
return;
|
||||
}
|
||||
packageInfo = pm.getPackageInfo(
|
||||
packageName, PackageManager.GET_ACTIVITIES);
|
||||
} catch (NameNotFoundException e) {
|
||||
Log.e(TAG, "cannot get package info for " + packageName, e);
|
||||
}
|
||||
SamplingProfilerIntegration.writeSnapshot(mBoundApplication.processName, packageInfo);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private class Idler implements MessageQueue.IdleHandler {
|
||||
@@ -6123,7 +6093,6 @@ public final class ActivityThread {
|
||||
|
||||
public static void main(String[] args) {
|
||||
Trace.traceBegin(Trace.TRACE_TAG_ACTIVITY_MANAGER, "ActivityThreadMain");
|
||||
SamplingProfilerIntegration.start();
|
||||
|
||||
// CloseGuard defaults to true and can be quite spammy. We
|
||||
// disable it here, but selectively enable it later (via
|
||||
|
||||
@@ -1,226 +0,0 @@
|
||||
/*
|
||||
* Copyright (C) 2009 The Android Open Source Project
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
package com.android.internal.os;
|
||||
|
||||
import android.content.pm.PackageInfo;
|
||||
import android.os.Build;
|
||||
import android.os.SystemProperties;
|
||||
import android.util.Log;
|
||||
import dalvik.system.profiler.BinaryHprofWriter;
|
||||
import dalvik.system.profiler.SamplingProfiler;
|
||||
import java.io.BufferedOutputStream;
|
||||
import java.io.File;
|
||||
import java.io.FileOutputStream;
|
||||
import java.io.IOException;
|
||||
import java.io.OutputStream;
|
||||
import java.io.PrintStream;
|
||||
import java.util.Date;
|
||||
import java.util.concurrent.Executor;
|
||||
import java.util.concurrent.Executors;
|
||||
import java.util.concurrent.ThreadFactory;
|
||||
import java.util.concurrent.atomic.AtomicBoolean;
|
||||
import libcore.io.IoUtils;
|
||||
|
||||
/**
|
||||
* Integrates the framework with Dalvik's sampling profiler.
|
||||
*/
|
||||
public class SamplingProfilerIntegration {
|
||||
|
||||
private static final String TAG = "SamplingProfilerIntegration";
|
||||
|
||||
public static final String SNAPSHOT_DIR = "/data/snapshots";
|
||||
|
||||
private static final boolean enabled;
|
||||
private static final Executor snapshotWriter;
|
||||
private static final int samplingProfilerMilliseconds;
|
||||
private static final int samplingProfilerDepth;
|
||||
|
||||
/** Whether or not a snapshot is being persisted. */
|
||||
private static final AtomicBoolean pending = new AtomicBoolean(false);
|
||||
|
||||
static {
|
||||
samplingProfilerMilliseconds = SystemProperties.getInt("persist.sys.profiler_ms", 0);
|
||||
samplingProfilerDepth = SystemProperties.getInt("persist.sys.profiler_depth", 4);
|
||||
if (samplingProfilerMilliseconds > 0) {
|
||||
File dir = new File(SNAPSHOT_DIR);
|
||||
dir.mkdirs();
|
||||
// the directory needs to be writable to anybody to allow file writing
|
||||
dir.setWritable(true, false);
|
||||
// the directory needs to be executable to anybody to allow file creation
|
||||
dir.setExecutable(true, false);
|
||||
if (dir.isDirectory()) {
|
||||
snapshotWriter = Executors.newSingleThreadExecutor(new ThreadFactory() {
|
||||
public Thread newThread(Runnable r) {
|
||||
return new Thread(r, TAG);
|
||||
}
|
||||
});
|
||||
enabled = true;
|
||||
Log.i(TAG, "Profiling enabled. Sampling interval ms: "
|
||||
+ samplingProfilerMilliseconds);
|
||||
} else {
|
||||
snapshotWriter = null;
|
||||
enabled = true;
|
||||
Log.w(TAG, "Profiling setup failed. Could not create " + SNAPSHOT_DIR);
|
||||
}
|
||||
} else {
|
||||
snapshotWriter = null;
|
||||
enabled = false;
|
||||
Log.i(TAG, "Profiling disabled.");
|
||||
}
|
||||
}
|
||||
|
||||
private static SamplingProfiler samplingProfiler;
|
||||
private static long startMillis;
|
||||
|
||||
/**
|
||||
* Is profiling enabled?
|
||||
*/
|
||||
public static boolean isEnabled() {
|
||||
return enabled;
|
||||
}
|
||||
|
||||
/**
|
||||
* Starts the profiler if profiling is enabled.
|
||||
*/
|
||||
public static void start() {
|
||||
if (!enabled) {
|
||||
return;
|
||||
}
|
||||
if (samplingProfiler != null) {
|
||||
Log.e(TAG, "SamplingProfilerIntegration already started at " + new Date(startMillis));
|
||||
return;
|
||||
}
|
||||
|
||||
ThreadGroup group = Thread.currentThread().getThreadGroup();
|
||||
SamplingProfiler.ThreadSet threadSet = SamplingProfiler.newThreadGroupThreadSet(group);
|
||||
samplingProfiler = new SamplingProfiler(samplingProfilerDepth, threadSet);
|
||||
samplingProfiler.start(samplingProfilerMilliseconds);
|
||||
startMillis = System.currentTimeMillis();
|
||||
}
|
||||
|
||||
/**
|
||||
* Writes a snapshot if profiling is enabled.
|
||||
*/
|
||||
public static void writeSnapshot(final String processName, final PackageInfo packageInfo) {
|
||||
if (!enabled) {
|
||||
return;
|
||||
}
|
||||
if (samplingProfiler == null) {
|
||||
Log.e(TAG, "SamplingProfilerIntegration is not started");
|
||||
return;
|
||||
}
|
||||
|
||||
/*
|
||||
* If we're already writing a snapshot, don't bother enqueueing another
|
||||
* request right now. This will reduce the number of individual
|
||||
* snapshots and in turn the total amount of memory consumed (one big
|
||||
* snapshot is smaller than N subset snapshots).
|
||||
*/
|
||||
if (pending.compareAndSet(false, true)) {
|
||||
snapshotWriter.execute(new Runnable() {
|
||||
public void run() {
|
||||
try {
|
||||
writeSnapshotFile(processName, packageInfo);
|
||||
} finally {
|
||||
pending.set(false);
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Writes the zygote's snapshot to internal storage if profiling is enabled.
|
||||
*/
|
||||
public static void writeZygoteSnapshot() {
|
||||
if (!enabled) {
|
||||
return;
|
||||
}
|
||||
writeSnapshotFile("zygote", null);
|
||||
samplingProfiler.shutdown();
|
||||
samplingProfiler = null;
|
||||
startMillis = 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* pass in PackageInfo to retrieve various values for snapshot header
|
||||
*/
|
||||
private static void writeSnapshotFile(String processName, PackageInfo packageInfo) {
|
||||
if (!enabled) {
|
||||
return;
|
||||
}
|
||||
samplingProfiler.stop();
|
||||
|
||||
/*
|
||||
* We use the global start time combined with the process name
|
||||
* as a unique ID. We can't use a counter because processes
|
||||
* restart. This could result in some overlap if we capture
|
||||
* two snapshots in rapid succession.
|
||||
*/
|
||||
String name = processName.replaceAll(":", ".");
|
||||
String path = SNAPSHOT_DIR + "/" + name + "-" + startMillis + ".snapshot";
|
||||
long start = System.currentTimeMillis();
|
||||
OutputStream outputStream = null;
|
||||
try {
|
||||
outputStream = new BufferedOutputStream(new FileOutputStream(path));
|
||||
PrintStream out = new PrintStream(outputStream);
|
||||
generateSnapshotHeader(name, packageInfo, out);
|
||||
if (out.checkError()) {
|
||||
throw new IOException();
|
||||
}
|
||||
BinaryHprofWriter.write(samplingProfiler.getHprofData(), outputStream);
|
||||
} catch (IOException e) {
|
||||
Log.e(TAG, "Error writing snapshot to " + path, e);
|
||||
return;
|
||||
} finally {
|
||||
IoUtils.closeQuietly(outputStream);
|
||||
}
|
||||
// set file readable to the world so that SamplingProfilerService
|
||||
// can put it to dropbox
|
||||
new File(path).setReadable(true, false);
|
||||
|
||||
long elapsed = System.currentTimeMillis() - start;
|
||||
Log.i(TAG, "Wrote snapshot " + path + " in " + elapsed + "ms.");
|
||||
samplingProfiler.start(samplingProfilerMilliseconds);
|
||||
}
|
||||
|
||||
/**
|
||||
* generate header for snapshots, with the following format
|
||||
* (like an HTTP header but without the \r):
|
||||
*
|
||||
* Version: <version number of profiler>\n
|
||||
* Process: <process name>\n
|
||||
* Package: <package name, if exists>\n
|
||||
* Package-Version: <version number of the package, if exists>\n
|
||||
* Build: <fingerprint>\n
|
||||
* \n
|
||||
* <the actual snapshot content begins here...>
|
||||
*/
|
||||
private static void generateSnapshotHeader(String processName, PackageInfo packageInfo,
|
||||
PrintStream out) {
|
||||
// profiler version
|
||||
out.println("Version: 3");
|
||||
out.println("Process: " + processName);
|
||||
if (packageInfo != null) {
|
||||
out.println("Package: " + packageInfo.packageName);
|
||||
out.println("Package-Version: " + packageInfo.versionCode);
|
||||
}
|
||||
out.println("Build: " + Build.FINGERPRINT);
|
||||
// single blank line means the end of snapshot header.
|
||||
out.println();
|
||||
}
|
||||
}
|
||||
@@ -648,8 +648,6 @@ public class ZygoteInit {
|
||||
try {
|
||||
Trace.traceBegin(Trace.TRACE_TAG_DALVIK, "ZygoteInit");
|
||||
RuntimeInit.enableDdms();
|
||||
// Start profiling the zygote initialization.
|
||||
SamplingProfilerIntegration.start();
|
||||
|
||||
boolean startSystemServer = false;
|
||||
String socketName = "zygote";
|
||||
@@ -679,9 +677,6 @@ public class ZygoteInit {
|
||||
SystemClock.uptimeMillis());
|
||||
Trace.traceEnd(Trace.TRACE_TAG_DALVIK);
|
||||
|
||||
// Finish profiling the zygote initialization.
|
||||
SamplingProfilerIntegration.writeZygoteSnapshot();
|
||||
|
||||
// Do an initial gc to clean up after startup
|
||||
Trace.traceBegin(Trace.TRACE_TAG_DALVIK, "PostZygoteInitGC");
|
||||
gcAndFinalize();
|
||||
|
||||
@@ -1,121 +0,0 @@
|
||||
/*
|
||||
* Copyright (C) 2010 The Android Open Source Project
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
package com.android.server;
|
||||
|
||||
import android.content.ContentResolver;
|
||||
import android.os.DropBoxManager;
|
||||
import android.os.FileObserver;
|
||||
import android.os.Binder;
|
||||
|
||||
import android.util.Slog;
|
||||
import android.content.Context;
|
||||
import android.database.ContentObserver;
|
||||
import android.os.SystemProperties;
|
||||
import android.provider.Settings;
|
||||
import com.android.internal.os.SamplingProfilerIntegration;
|
||||
|
||||
import java.io.File;
|
||||
import java.io.FileDescriptor;
|
||||
import java.io.IOException;
|
||||
import java.io.PrintWriter;
|
||||
|
||||
public class SamplingProfilerService extends Binder {
|
||||
|
||||
private static final String TAG = "SamplingProfilerService";
|
||||
private static final boolean LOCAL_LOGV = false;
|
||||
public static final String SNAPSHOT_DIR = SamplingProfilerIntegration.SNAPSHOT_DIR;
|
||||
|
||||
private final Context mContext;
|
||||
private FileObserver snapshotObserver;
|
||||
|
||||
public SamplingProfilerService(Context context) {
|
||||
mContext = context;
|
||||
registerSettingObserver(context);
|
||||
startWorking(context);
|
||||
}
|
||||
|
||||
private void startWorking(Context context) {
|
||||
if (LOCAL_LOGV) Slog.v(TAG, "starting SamplingProfilerService!");
|
||||
|
||||
final DropBoxManager dropbox =
|
||||
(DropBoxManager) context.getSystemService(Context.DROPBOX_SERVICE);
|
||||
|
||||
// before FileObserver is ready, there could have already been some snapshots
|
||||
// in the directory, we don't want to miss them
|
||||
File[] snapshotFiles = new File(SNAPSHOT_DIR).listFiles();
|
||||
for (int i = 0; snapshotFiles != null && i < snapshotFiles.length; i++) {
|
||||
handleSnapshotFile(snapshotFiles[i], dropbox);
|
||||
}
|
||||
|
||||
// detect new snapshot and put it in dropbox
|
||||
// delete it afterwards no matter what happened before
|
||||
// Note: needs listening at event ATTRIB rather than CLOSE_WRITE, because we set the
|
||||
// readability of snapshot files after writing them!
|
||||
snapshotObserver = new FileObserver(SNAPSHOT_DIR, FileObserver.ATTRIB) {
|
||||
@Override
|
||||
public void onEvent(int event, String path) {
|
||||
handleSnapshotFile(new File(SNAPSHOT_DIR, path), dropbox);
|
||||
}
|
||||
};
|
||||
snapshotObserver.startWatching();
|
||||
|
||||
if (LOCAL_LOGV) Slog.v(TAG, "SamplingProfilerService activated");
|
||||
}
|
||||
|
||||
private void handleSnapshotFile(File file, DropBoxManager dropbox) {
|
||||
try {
|
||||
dropbox.addFile(TAG, file, 0);
|
||||
if (LOCAL_LOGV) Slog.v(TAG, file.getPath() + " added to dropbox");
|
||||
} catch (IOException e) {
|
||||
Slog.e(TAG, "Can't add " + file.getPath() + " to dropbox", e);
|
||||
} finally {
|
||||
file.delete();
|
||||
}
|
||||
}
|
||||
|
||||
private void registerSettingObserver(Context context) {
|
||||
ContentResolver contentResolver = context.getContentResolver();
|
||||
contentResolver.registerContentObserver(
|
||||
Settings.Global.getUriFor(Settings.Global.SAMPLING_PROFILER_MS),
|
||||
false, new SamplingProfilerSettingsObserver(contentResolver));
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void dump(FileDescriptor fd, PrintWriter pw, String[] args) {
|
||||
mContext.enforceCallingOrSelfPermission(android.Manifest.permission.DUMP, TAG);
|
||||
|
||||
pw.println("SamplingProfilerService:");
|
||||
pw.println("Watching directory: " + SNAPSHOT_DIR);
|
||||
}
|
||||
|
||||
private class SamplingProfilerSettingsObserver extends ContentObserver {
|
||||
private ContentResolver mContentResolver;
|
||||
public SamplingProfilerSettingsObserver(ContentResolver contentResolver) {
|
||||
super(null);
|
||||
mContentResolver = contentResolver;
|
||||
onChange(false);
|
||||
}
|
||||
@Override
|
||||
public void onChange(boolean selfChange) {
|
||||
Integer samplingProfilerMs = Settings.Global.getInt(
|
||||
mContentResolver, Settings.Global.SAMPLING_PROFILER_MS, 0);
|
||||
// setting this secure property will start or stop sampling profiler,
|
||||
// as well as adjust the the time between taking snapshots.
|
||||
SystemProperties.set("persist.sys.profiler_ms", samplingProfilerMs.toString());
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -51,7 +51,6 @@ import android.view.WindowManager;
|
||||
import com.android.internal.R;
|
||||
import com.android.internal.app.NightDisplayController;
|
||||
import com.android.internal.os.BinderInternal;
|
||||
import com.android.internal.os.SamplingProfilerIntegration;
|
||||
import com.android.internal.os.ZygoteInit;
|
||||
import com.android.internal.policy.EmergencyAffordanceManager;
|
||||
import com.android.internal.widget.ILockSettings;
|
||||
@@ -281,18 +280,6 @@ public final class SystemServer {
|
||||
// the property. http://b/11463182
|
||||
SystemProperties.set("persist.sys.dalvik.vm.lib.2", VMRuntime.getRuntime().vmLibrary());
|
||||
|
||||
// Enable the sampling profiler.
|
||||
if (SamplingProfilerIntegration.isEnabled()) {
|
||||
SamplingProfilerIntegration.start();
|
||||
mProfilerSnapshotTimer = new Timer();
|
||||
mProfilerSnapshotTimer.schedule(new TimerTask() {
|
||||
@Override
|
||||
public void run() {
|
||||
SamplingProfilerIntegration.writeSnapshot("system_server", null);
|
||||
}
|
||||
}, SNAPSHOT_INTERVAL, SNAPSHOT_INTERVAL);
|
||||
}
|
||||
|
||||
// Mmmmmm... more memory!
|
||||
VMRuntime.getRuntime().clearGrowthLimit();
|
||||
|
||||
@@ -568,8 +555,6 @@ public final class SystemServer {
|
||||
false);
|
||||
boolean disableTextServices = SystemProperties.getBoolean("config.disable_textservices",
|
||||
false);
|
||||
boolean disableSamplingProfiler = SystemProperties.getBoolean("config.disable_samplingprof",
|
||||
false);
|
||||
boolean disableConsumerIr = SystemProperties.getBoolean("config.disable_consumerir", false);
|
||||
boolean disableVrManager = SystemProperties.getBoolean("config.disable_vrmanager", false);
|
||||
boolean disableCameraService = SystemProperties.getBoolean("config.disable_cameraservice",
|
||||
@@ -1088,21 +1073,6 @@ public final class SystemServer {
|
||||
}
|
||||
Trace.traceEnd(Trace.TRACE_TAG_SYSTEM_SERVER);
|
||||
|
||||
if (!disableSamplingProfiler) {
|
||||
traceBeginAndSlog("StartSamplingProfilerService");
|
||||
try {
|
||||
// need to add this service even if SamplingProfilerIntegration.isEnabled()
|
||||
// is false, because it is this service that detects system property change and
|
||||
// turns on SamplingProfilerIntegration. Plus, when sampling profiler doesn't work,
|
||||
// there is little overhead for running this service.
|
||||
ServiceManager.addService("samplingprofiler",
|
||||
new SamplingProfilerService(context));
|
||||
} catch (Throwable e) {
|
||||
reportWtf("starting SamplingProfiler Service", e);
|
||||
}
|
||||
Trace.traceEnd(Trace.TRACE_TAG_SYSTEM_SERVER);
|
||||
}
|
||||
|
||||
if (!disableNetwork && !disableNetworkTime) {
|
||||
traceBeginAndSlog("StartNetworkTimeUpdateService");
|
||||
try {
|
||||
|
||||
Reference in New Issue
Block a user