blob: e769907585e5412df8c413906bb278dc0494bd37 [file] [log] [blame]
/*
* Main entry of app process.
*
* Starts the interpreted runtime, then starts up the application.
*
*/
#define LOG_TAG "appproc"
#include "class_loader.h"
#include "jni_internal.h"
#include "stringprintf.h"
#include "thread.h"
#include <binder/IPCThreadState.h>
#include <binder/ProcessState.h>
#include <utils/Log.h>
#include <cutils/process_name.h>
#include <cutils/memory.h>
#include <android_runtime/AndroidRuntime.h>
#include <stdio.h>
#include <unistd.h>
namespace android {
void app_usage()
{
fprintf(stderr,
"Usage: oat_process [java-options] cmd-dir start-class-name [options]\n");
}
class AppRuntime : public AndroidRuntime
{
public:
AppRuntime()
: mParentDir(NULL)
, mClassName(NULL)
, mClass(NULL)
, mArgC(0)
, mArgV(NULL)
{
}
#if 0
// this appears to be unused
const char* getParentDir() const
{
return mParentDir;
}
#endif
const char* getClassName() const
{
return mClassName;
}
virtual void onVmCreated(JNIEnv* env)
{
if (mClassName == NULL) {
return; // Zygote. Nothing to do here.
}
/*
* This is a little awkward because the JNI FindClass call uses the
* class loader associated with the native method we're executing in.
* If called in onStarted (from RuntimeInit.finishInit because we're
* launching "am", for example), FindClass would see that we're calling
* from a boot class' native method, and so wouldn't look for the class
* we're trying to look up in CLASSPATH. Unfortunately it needs to,
* because the "am" classes are not boot classes.
*
* The easiest fix is to call FindClass here, early on before we start
* executing boot class Java code and thereby deny ourselves access to
* non-boot classes.
*/
char* slashClassName = toSlashClassName(mClassName);
mClass = env->FindClass(slashClassName);
if (mClass == NULL) {
LOG(ERROR) << StringPrintf("ERROR: could not find class '%s'\n", mClassName);
}
free(slashClassName);
mClass = reinterpret_cast<jclass>(env->NewGlobalRef(mClass));
// TODO: remove this ClassLoader code
jclass ApplicationLoaders = env->FindClass("android/app/ApplicationLoaders");
jmethodID getDefault = env->GetStaticMethodID(ApplicationLoaders,
"getDefault",
"()Landroid/app/ApplicationLoaders;");
jfieldID mLoaders = env->GetFieldID(ApplicationLoaders, "mLoaders", "Ljava/util/Map;");
jclass BootClassLoader = env->FindClass("java/lang/BootClassLoader");
jmethodID getInstance = env->GetStaticMethodID(BootClassLoader,
"getInstance",
"()Ljava/lang/BootClassLoader;");
jclass ClassLoader = env->FindClass("java/lang/ClassLoader");
jfieldID parent = env->GetFieldID(ClassLoader, "parent", "Ljava/lang/ClassLoader;");
jclass Map = env->FindClass("java/util/Map");
jmethodID put = env->GetMethodID(Map,
"put",
"(Ljava/lang/Object;Ljava/lang/Object;)Ljava/lang/Object;");
jclass BaseDexClassLoader = env->FindClass("dalvik/system/BaseDexClassLoader");
jfieldID originalPath = env->GetFieldID(BaseDexClassLoader, "originalPath", "Ljava/lang/String;");
jfieldID pathList = env->GetFieldID(BaseDexClassLoader, "pathList", "Ldalvik/system/DexPathList;");
jclass DexPathList = env->FindClass("dalvik/system/DexPathList");
jmethodID init = env->GetMethodID(DexPathList,
"<init>",
"(Ljava/lang/ClassLoader;Ljava/lang/String;Ljava/lang/String;Ljava/io/File;)V");
// Set the parent of our pre-existing ClassLoader to the non-null BootClassLoader.getInstance()
const art::ClassLoader* class_loader_object = art::Thread::Current()->GetClassLoaderOverride();
jobject class_loader = art::AddLocalReference<jobject>(env, class_loader_object);
jobject boot_class_loader = env->CallStaticObjectMethod(BootClassLoader, getInstance);
env->SetObjectField(class_loader, parent, boot_class_loader);
// Create a DexPathList
jstring dex_path = env->NewStringUTF("/system/app/Calculator.apk");
jstring library_path = env->NewStringUTF("/data/data/com.android.calculator2/lib");
jobject dex_path_list = env->NewObject(DexPathList, init,
boot_class_loader, dex_path, library_path, NULL);
// Set DexPathList into our pre-existing ClassLoader
env->SetObjectField(class_loader, pathList, dex_path_list);
env->SetObjectField(class_loader, originalPath, dex_path);
// Stash our pre-existing ClassLoader into ApplicationLoaders.getDefault().mLoaders
// under the expected name.
jobject application_loaders = env->CallStaticObjectMethod(ApplicationLoaders, getDefault);
jobject loaders = env->GetObjectField(application_loaders, mLoaders);
env->CallObjectMethod(loaders, put, dex_path, class_loader);
}
virtual void onStarted()
{
sp<ProcessState> proc = ProcessState::self();
LOGV("App process: starting thread pool.\n");
proc->startThreadPool();
AndroidRuntime* ar = AndroidRuntime::getRuntime();
ar->callMain(mClassName, mClass, mArgC, mArgV);
IPCThreadState::self()->stopProcess();
}
virtual void onZygoteInit()
{
sp<ProcessState> proc = ProcessState::self();
LOGV("App process: starting thread pool.\n");
proc->startThreadPool();
}
virtual void onExit(int code)
{
if (mClassName == NULL) {
// if zygote
IPCThreadState::self()->stopProcess();
}
AndroidRuntime::onExit(code);
}
const char* mParentDir;
const char* mClassName;
jclass mClass;
int mArgC;
const char* const* mArgV;
};
}
using namespace android;
/*
* sets argv0 to as much of newArgv0 as will fit
*/
static void setArgv0(const char *argv0, const char *newArgv0)
{
strlcpy(const_cast<char *>(argv0), newArgv0, strlen(argv0));
}
int main(int argc, const char* argv[])
{
// These are global variables in ProcessState.cpp
mArgC = argc;
mArgV = argv;
mArgLen = 0;
for (int i=0; i<argc; i++) {
mArgLen += strlen(argv[i]) + 1;
}
mArgLen--;
AppRuntime runtime;
const char* argv0 = argv[0];
// Process command line arguments
// ignore argv[0]
argc--;
argv++;
// ignore /system/bin/app_process when invoked via WrapperInit
if (strcmp(argv[0], "/system/bin/app_process") == 0) {
LOG(INFO) << "Removing /system/bin/app_process argument";
argc--;
argv++;
for (int i = 0; i < argc; i++) {
LOG(INFO) << StringPrintf("argv[%d]=%s", i, argv[i]);
}
}
// TODO: remove Calculator special case
int oatArgc = argc + 2;
const char* oatArgv[oatArgc];
if (strcmp(argv[0], "-Ximage:/system/framework/boot.art") != 0) {
LOG(INFO) << "Adding oat arguments";
oatArgv[0] = "-Ximage:/system/framework/boot.art";
oatArgv[1] = "-Ximage:/system/app/Calculator.art";
setenv("CLASSPATH", "/system/app/Calculator.apk", 1);
memcpy(oatArgv + (oatArgc - argc), argv, argc * sizeof(*argv));
argv = oatArgv;
argc = oatArgc;
for (int i = 0; i < argc; i++) {
LOG(INFO) << StringPrintf("argv[%d]=%s", i, argv[i]);
}
}
// TODO: remove the heap arguments when implicit garbage collection enabled
LOG(INFO) << "Adding heap arguments";
int heapArgc = argc + 2;
const char* heapArgv[heapArgc];
heapArgv[0] = "-Xms64m";
heapArgv[1] = "-Xmx64m";
memcpy(heapArgv + (heapArgc - argc), argv, argc * sizeof(*argv));
argv = heapArgv;
argc = heapArgc;
for (int i = 0; i < argc; i++) {
LOG(INFO) << StringPrintf("argv[%d]=%s", i, argv[i]);
}
// TODO: change the system default to not perform preloading
LOG(INFO) << "Disabling preloading";
for (int i = 0; i < argc; i++) {
if (strcmp(argv[i], "preload") == 0) {
argv[i] = "nopreload";
break;
}
}
for (int i = 0; i < argc; i++) {
LOG(INFO) << StringPrintf("argv[%d]=%s", i, argv[i]);
}
// Everything up to '--' or first non '-' arg goes to the vm
int i = runtime.addVmArguments(argc, argv);
// Parse runtime arguments. Stop at first unrecognized option.
bool zygote = false;
bool startSystemServer = false;
bool application = false;
const char* parentDir = NULL;
const char* niceName = NULL;
const char* className = NULL;
while (i < argc) {
const char* arg = argv[i++];
if (!parentDir) {
parentDir = arg;
} else if (strcmp(arg, "--zygote") == 0) {
zygote = true;
niceName = "zygote";
} else if (strcmp(arg, "--start-system-server") == 0) {
startSystemServer = true;
} else if (strcmp(arg, "--application") == 0) {
application = true;
} else if (strncmp(arg, "--nice-name=", 12) == 0) {
niceName = arg + 12;
} else {
className = arg;
break;
}
}
if (niceName && *niceName) {
setArgv0(argv0, niceName);
set_process_name(niceName);
}
runtime.mParentDir = parentDir;
if (zygote) {
runtime.start("com.android.internal.os.ZygoteInit",
startSystemServer ? "start-system-server" : "");
} else if (className) {
// Remainder of args get passed to startup class main()
runtime.mClassName = className;
runtime.mArgC = argc - i;
runtime.mArgV = argv + i;
runtime.start("com.android.internal.os.RuntimeInit",
application ? "application" : "tool");
} else {
fprintf(stderr, "Error: no class name or --zygote supplied.\n");
app_usage();
LOG(FATAL) << "oat_process: no class name or --zygote supplied.";
return 10;
}
}