The monkey now gets told about system hangs, and does stuff about it! Most especially, it can collect a bug report. Monkey wins! Change-Id: Ic8350721e715ef21d1ec813c6aff9a83262d5faa
245 lines
9.5 KiB
Java
245 lines
9.5 KiB
Java
/*
|
|
* 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.development;
|
|
|
|
import android.app.Activity;
|
|
import android.app.ActivityManagerNative;
|
|
import android.app.IActivityController;
|
|
import android.app.IActivityManager;
|
|
import android.app.Service;
|
|
import android.content.BroadcastReceiver;
|
|
import android.content.Context;
|
|
import android.content.Intent;
|
|
import android.content.IntentFilter;
|
|
import android.os.Bundle;
|
|
import android.os.IBinder;
|
|
import android.os.IPowerManager;
|
|
import android.os.Process;
|
|
import android.os.RemoteException;
|
|
import android.os.ServiceManager;
|
|
import android.util.Log;
|
|
import android.view.View;
|
|
import android.widget.Button;
|
|
|
|
public class BadBehaviorActivity extends Activity {
|
|
private static final String TAG = "BadBehaviorActivity";
|
|
|
|
private static class BadBehaviorException extends RuntimeException {
|
|
BadBehaviorException() {
|
|
super("Whatcha gonna do, whatcha gonna do",
|
|
new IllegalStateException("When they come for you"));
|
|
}
|
|
}
|
|
|
|
public static class BadReceiver extends BroadcastReceiver {
|
|
@Override
|
|
public void onReceive(Context context, Intent intent) {
|
|
Log.i(TAG, "in broadcast receiver -- about to hang");
|
|
try { Thread.sleep(20000); } catch (InterruptedException e) { Log.wtf(TAG, e); }
|
|
Log.i(TAG, "broadcast receiver hang finished -- returning");
|
|
}
|
|
};
|
|
|
|
public static class BadService extends Service {
|
|
@Override
|
|
public IBinder onBind(Intent intent) {
|
|
return null;
|
|
}
|
|
|
|
@Override
|
|
public int onStartCommand(Intent intent, int flags, int id) {
|
|
Log.i(TAG, "in service start -- about to hang");
|
|
try { Thread.sleep(30000); } catch (InterruptedException e) { Log.wtf(TAG, e); }
|
|
Log.i(TAG, "service hang finished -- stopping and returning");
|
|
stopSelf();
|
|
return START_NOT_STICKY;
|
|
}
|
|
}
|
|
|
|
public static class BadController extends IActivityController.Stub {
|
|
private int mDelay;
|
|
|
|
public BadController(int delay) { mDelay = delay; }
|
|
|
|
public boolean activityStarting(Intent intent, String pkg) {
|
|
try {
|
|
ActivityManagerNative.getDefault().setActivityController(null);
|
|
} catch (RemoteException e) {
|
|
Log.e(TAG, "Can't call IActivityManager.setActivityController", e);
|
|
}
|
|
|
|
if (mDelay > 0) {
|
|
Log.i(TAG, "in activity controller -- about to hang");
|
|
try { Thread.sleep(mDelay); } catch (InterruptedException e) { Log.wtf(TAG, e); }
|
|
Log.i(TAG, "activity controller hang finished -- disabling and returning");
|
|
mDelay = 0;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
public boolean activityResuming(String pkg) {
|
|
return true;
|
|
}
|
|
|
|
public boolean appCrashed(String proc, int pid, String m, String m2, long time, String st) {
|
|
return true;
|
|
}
|
|
|
|
public int appEarlyNotResponding(String processName, int pid, String annotation) {
|
|
return 0;
|
|
}
|
|
|
|
public int appNotResponding(String proc, int pid, String st) {
|
|
return 0;
|
|
}
|
|
|
|
public int systemNotResponding(String message) {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onCreate(Bundle icicle) {
|
|
super.onCreate(icicle);
|
|
|
|
if (getIntent().getBooleanExtra("anr", false)) {
|
|
Log.i(TAG, "in ANR activity -- about to hang");
|
|
try { Thread.sleep(20000); } catch (InterruptedException e) { Log.wtf(TAG, e); }
|
|
Log.i(TAG, "activity hang finished -- finishing");
|
|
finish();
|
|
return;
|
|
}
|
|
|
|
if (getIntent().getBooleanExtra("dummy", false)) {
|
|
Log.i(TAG, "in dummy activity -- finishing");
|
|
finish();
|
|
return;
|
|
}
|
|
|
|
setContentView(R.layout.bad_behavior);
|
|
|
|
Button crash_system = (Button) findViewById(R.id.bad_behavior_crash_system);
|
|
crash_system.setOnClickListener(new View.OnClickListener() {
|
|
public void onClick(View v) {
|
|
try {
|
|
IBinder b = ServiceManager.getService(POWER_SERVICE);
|
|
IPowerManager pm = IPowerManager.Stub.asInterface(b);
|
|
pm.crash("Crashed by BadBehaviorActivity");
|
|
} catch (RemoteException e) {
|
|
Log.e(TAG, "Can't call IPowerManager.crash()", e);
|
|
}
|
|
}
|
|
});
|
|
|
|
Button crash_main = (Button) findViewById(R.id.bad_behavior_crash_main);
|
|
crash_main.setOnClickListener(new View.OnClickListener() {
|
|
public void onClick(View v) { throw new BadBehaviorException(); }
|
|
});
|
|
|
|
Button crash_thread = (Button) findViewById(R.id.bad_behavior_crash_thread);
|
|
crash_thread.setOnClickListener(new View.OnClickListener() {
|
|
public void onClick(View v) {
|
|
new Thread() {
|
|
@Override
|
|
public void run() { throw new BadBehaviorException(); }
|
|
}.start();
|
|
}
|
|
});
|
|
|
|
Button crash_native = (Button) findViewById(R.id.bad_behavior_crash_native);
|
|
crash_native.setOnClickListener(new View.OnClickListener() {
|
|
public void onClick(View v) {
|
|
// For some reason, the JVM needs two of these to get the hint
|
|
Log.i(TAG, "Native crash pressed -- about to kill -11 self");
|
|
Process.sendSignal(Process.myPid(), 11);
|
|
Process.sendSignal(Process.myPid(), 11);
|
|
Log.i(TAG, "Finished kill -11, should be dead or dying");
|
|
}
|
|
});
|
|
|
|
Button wtf = (Button) findViewById(R.id.bad_behavior_wtf);
|
|
wtf.setOnClickListener(new View.OnClickListener() {
|
|
public void onClick(View v) { Log.wtf(TAG, "Apps Behaving Badly"); }
|
|
});
|
|
|
|
Button anr = (Button) findViewById(R.id.bad_behavior_anr);
|
|
anr.setOnClickListener(new View.OnClickListener() {
|
|
public void onClick(View v) {
|
|
Log.i(TAG, "ANR pressed -- about to hang");
|
|
try { Thread.sleep(20000); } catch (InterruptedException e) { Log.wtf(TAG, e); }
|
|
Log.i(TAG, "hang finished -- returning");
|
|
}
|
|
});
|
|
|
|
Button anr_activity = (Button) findViewById(R.id.bad_behavior_anr_activity);
|
|
anr_activity.setOnClickListener(new View.OnClickListener() {
|
|
public void onClick(View v) {
|
|
Intent intent = new Intent(BadBehaviorActivity.this, BadBehaviorActivity.class);
|
|
Log.i(TAG, "ANR activity pressed -- about to launch");
|
|
startActivity(intent.putExtra("anr", true));
|
|
}
|
|
});
|
|
|
|
Button anr_broadcast = (Button) findViewById(R.id.bad_behavior_anr_broadcast);
|
|
anr_broadcast.setOnClickListener(new View.OnClickListener() {
|
|
public void onClick(View v) {
|
|
Log.i(TAG, "ANR broadcast pressed -- about to send");
|
|
sendOrderedBroadcast(new Intent("com.android.development.BAD_BEHAVIOR"), null);
|
|
}
|
|
});
|
|
|
|
Button anr_service = (Button) findViewById(R.id.bad_behavior_anr_service);
|
|
anr_service.setOnClickListener(new View.OnClickListener() {
|
|
public void onClick(View v) {
|
|
Log.i(TAG, "ANR service pressed -- about to start");
|
|
startService(new Intent(BadBehaviorActivity.this, BadService.class));
|
|
}
|
|
});
|
|
|
|
Button anr_system = (Button) findViewById(R.id.bad_behavior_anr_system);
|
|
anr_system.setOnClickListener(new View.OnClickListener() {
|
|
public void onClick(View v) {
|
|
Intent intent = new Intent(BadBehaviorActivity.this, BadBehaviorActivity.class);
|
|
Log.i(TAG, "ANR system pressed -- about to engage");
|
|
try {
|
|
ActivityManagerNative.getDefault().setActivityController(
|
|
new BadController(20000));
|
|
} catch (RemoteException e) {
|
|
Log.e(TAG, "Can't call IActivityManager.setActivityController", e);
|
|
}
|
|
startActivity(intent.putExtra("dummy", true));
|
|
}
|
|
});
|
|
|
|
Button wedge_system = (Button) findViewById(R.id.bad_behavior_wedge_system);
|
|
wedge_system.setOnClickListener(new View.OnClickListener() {
|
|
public void onClick(View v) {
|
|
Intent intent = new Intent(BadBehaviorActivity.this, BadBehaviorActivity.class);
|
|
Log.i(TAG, "Wedge system pressed -- about to engage");
|
|
try {
|
|
ActivityManagerNative.getDefault().setActivityController(
|
|
new BadController(300000));
|
|
} catch (RemoteException e) {
|
|
Log.e(TAG, "Can't call IActivityManager.setActivityController", e);
|
|
}
|
|
startActivity(intent.putExtra("dummy", true));
|
|
}
|
|
});
|
|
}
|
|
}
|