Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(1137)

Unified Diff: mojo/runner/android/apk/src/org/chromium/mojo/shell/AndroidHandler.java

Issue 1149813002: Changes caching logic of mojo java apps (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: fix3 Created 5 years, 7 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: mojo/runner/android/apk/src/org/chromium/mojo/shell/AndroidHandler.java
diff --git a/mojo/runner/android/apk/src/org/chromium/mojo/shell/AndroidHandler.java b/mojo/runner/android/apk/src/org/chromium/mojo/shell/AndroidHandler.java
index 0ccc2ad18df6d986292d065f57456c15634fd042..1b730214f1d6aeaf18b73f3b6d1ea4a9aadd31b1 100644
--- a/mojo/runner/android/apk/src/org/chromium/mojo/shell/AndroidHandler.java
+++ b/mojo/runner/android/apk/src/org/chromium/mojo/shell/AndroidHandler.java
@@ -5,14 +5,15 @@
package org.chromium.mojo.shell;
import android.content.Context;
-import android.util.Log;
import dalvik.system.DexClassLoader;
import org.chromium.base.CalledByNative;
import org.chromium.base.JNINamespace;
+import org.chromium.base.Log;
import java.io.File;
+import java.io.FilenameFilter;
import java.io.IOException;
import java.lang.reflect.Constructor;
@@ -45,6 +46,10 @@ public class AndroidHandler {
private static final String APP_DIRECTORY = "applications";
private static final String ASSET_DIRECTORY = "assets";
+ private static final String INTERNAL_DIRECTORY = "internal";
+
+ private enum AppType { CACHED, UNCACHED }
+
/**
* Deletes directories holding the temporary files. This should be called early on shell startup
* to clean up after the previous run.
@@ -77,45 +82,158 @@ public class AndroidHandler {
@CalledByNative
private static boolean bootstrap(Context context, String archivePath, int handle,
long runApplicationPtr) {
- File bootstrap_java_library;
- File bootstrap_native_library;
+ File bootstrapJavaLibrary;
+ File bootstrapNativeLibrary;
try {
- bootstrap_java_library = FileHelper.extractFromAssets(context, BOOTSTRAP_JAVA_LIBRARY,
- getAssetDir(context), true);
- bootstrap_native_library = FileHelper.extractFromAssets(context,
- BOOTSTRAP_NATIVE_LIBRARY, getAssetDir(context), true);
+ bootstrapJavaLibrary = FileHelper.extractFromAssets(context, BOOTSTRAP_JAVA_LIBRARY,
+ getAssetDir(context), FileHelper.FileType.TEMPORARY);
+ bootstrapNativeLibrary = FileHelper.extractFromAssets(context, BOOTSTRAP_NATIVE_LIBRARY,
+ getAssetDir(context), FileHelper.FileType.TEMPORARY);
} catch (Exception e) {
Log.e(TAG, "Extraction of bootstrap files from assets failed.", e);
return false;
}
- File application_java_library;
- File application_native_library;
+ File applicationJavaLibrary;
+ File applicationNativeLibrary;
try {
File archive = new File(archivePath);
- application_java_library = FileHelper.extractFromArchive(archive, JAVA_LIBRARY_SUFFIX,
- getAppDir(context));
- application_native_library = FileHelper.extractFromArchive(archive,
- NATIVE_LIBRARY_SUFFIX, getAppDir(context));
+ applicationJavaLibrary =
+ FileHelper.extractFromArchive(archive, JAVA_LIBRARY_SUFFIX, getAppDir(context),
+ FileHelper.FileType.TEMPORARY, FileHelper.ArchiveType.NORMAL);
+ applicationNativeLibrary = FileHelper.extractFromArchive(archive, NATIVE_LIBRARY_SUFFIX,
+ getAppDir(context), FileHelper.FileType.TEMPORARY,
+ FileHelper.ArchiveType.NORMAL);
} catch (Exception e) {
Log.e(TAG, "Extraction of application files from the archive failed.", e);
return false;
}
- String dexPath = bootstrap_java_library.getAbsolutePath() + File.pathSeparator
- + application_java_library.getAbsolutePath();
- DexClassLoader bootstrapLoader = new DexClassLoader(dexPath,
- getDexOutputDir(context).getAbsolutePath(), null,
- ClassLoader.getSystemClassLoader());
+ return runApp(context, getDexOutputDir(context), applicationJavaLibrary,
+ applicationNativeLibrary, bootstrapJavaLibrary, bootstrapNativeLibrary, handle,
+ runApplicationPtr, AppType.UNCACHED);
+ }
+
+ private static File findFileInDirectoryMatchingSuffix(
+ File dir, final String suffix, final String ignore) {
+ File[] matchingFiles = dir.listFiles(new FilenameFilter() {
+ public boolean accept(File dir, String name) {
+ return !name.equals(ignore) && !name.equals(BOOTSTRAP_JAVA_LIBRARY)
+ && !name.equals(BOOTSTRAP_NATIVE_LIBRARY) && name.endsWith(suffix);
+ }
+ });
+ return matchingFiles != null && matchingFiles.length == 1 ? matchingFiles[0] : null;
+ }
+
+ /**
+ * Extracts and runs a cached application.
+ *
+ * @param context the application context
+ * @param archivePath the path of the archive containing the application to be run
+ * @param appPathString path where the cached app's resources and other files are
+ * @param handle handle to the shell to be passed to the native application. On the Java side
+ * this is opaque payload.
+ * @param runApplicationPtr pointer to the function that will set the native thunks and call
+ * into the application MojoMain. On the Java side this is opaque
+ * payload.
+ */
+ @CalledByNative
+ private static boolean bootstrapCachedApp(Context context, String archivePath,
+ String appPathString, int handle, long runApplicationPtr) {
+ final String appName = new File(appPathString).getName();
+ final File internalDir = new File(new File(appPathString), INTERNAL_DIRECTORY);
+ if (!internalDir.exists() && !internalDir.mkdirs()) {
+ Log.e(TAG, "Unable to create output dir " + internalDir.getAbsolutePath());
+ return false;
+ }
+ final File timestamp = FileHelper.prepareDirectoryForAssets(context, internalDir);
+ // We make the bootstrap library have a unique name on disk as otherwise we end up sharing
+ // bootstrap.so,
+ // which doesn't work.
+ // TODO(sky): figure out why android is caching the names.
+ final String bootstrapNativeLibraryName = appName + "-" + BOOTSTRAP_NATIVE_LIBRARY;
+ File bootstrapJavaLibrary = new File(internalDir, BOOTSTRAP_JAVA_LIBRARY);
+ File bootstrapNativeLibrary = new File(internalDir, bootstrapNativeLibraryName);
+ try {
+ // Use the files on disk if we have them, if not extract from the archive.
+ if (!bootstrapJavaLibrary.exists()) {
+ bootstrapJavaLibrary = FileHelper.extractFromAssets(context, BOOTSTRAP_JAVA_LIBRARY,
+ internalDir, FileHelper.FileType.PERMANENT);
+ }
+ if (!bootstrapNativeLibrary.exists()) {
+ final File extractedBootstrapNativeLibrary = FileHelper.extractFromAssets(context,
+ BOOTSTRAP_NATIVE_LIBRARY, internalDir, FileHelper.FileType.PERMANENT);
+ if (!extractedBootstrapNativeLibrary.renameTo(bootstrapNativeLibrary)) {
+ Log.e(TAG, "Unable to rename bootstrap library "
+ + bootstrapNativeLibrary.getAbsolutePath());
+ return false;
+ }
+ }
+ } catch (Exception e) {
+ Log.e(TAG, "Extraction of bootstrap files from assets failed.", e);
+ return false;
+ }
+
+ // Because we allow the .so and .dex.jar to be anything we have to search in the internal
+ // directory for matching files.
+ // If we find one, we assume it's the one we want.
+ File applicationJavaLibrary =
+ findFileInDirectoryMatchingSuffix(internalDir, JAVA_LIBRARY_SUFFIX, "");
+ File applicationNativeLibrary = findFileInDirectoryMatchingSuffix(
+ internalDir, NATIVE_LIBRARY_SUFFIX, bootstrapNativeLibraryName);
+ try {
+ File archive = new File(archivePath);
+ if (applicationJavaLibrary == null) {
+ applicationJavaLibrary = FileHelper.extractFromArchive(archive, JAVA_LIBRARY_SUFFIX,
+ internalDir, FileHelper.FileType.PERMANENT,
+ FileHelper.ArchiveType.CONTENT_HANDLER);
+ }
+ if (applicationNativeLibrary == null) {
+ applicationNativeLibrary = FileHelper.extractFromArchive(archive,
+ NATIVE_LIBRARY_SUFFIX, internalDir, FileHelper.FileType.PERMANENT,
+ FileHelper.ArchiveType.CONTENT_HANDLER);
+ }
+ } catch (Exception e) {
+ Log.e(TAG, "Extraction of application files from the archive failed.", e);
+ return false;
+ }
+
+ FileHelper.createTimestampIfNecessary(timestamp);
+
+ return runApp(context, new File(internalDir, DEX_OUTPUT_DIRECTORY), applicationJavaLibrary,
+ applicationNativeLibrary, bootstrapJavaLibrary, bootstrapNativeLibrary, handle,
+ runApplicationPtr, AppType.CACHED);
+ }
+
+ /**
+ * Creates the class loader containing the bootstrap classes and runs it.
+ *
+ * @return true if successfully ran, false if encounteres some sort of error.
+ */
+ private static boolean runApp(Context context, File dexOutputDir, File applicationJavaLibrary,
+ File applicationNativeLibrary, File bootstrapJavaLibrary, File bootstrapNativeLibrary,
+ int handle, long runApplicationPtr, AppType appType) {
+ final String dexPath = bootstrapJavaLibrary.getAbsolutePath() + File.pathSeparator
+ + applicationJavaLibrary.getAbsolutePath();
+ if (!dexOutputDir.exists() && !dexOutputDir.mkdirs()) {
+ Log.e(TAG, "Unable to create dex output dir " + dexOutputDir.getAbsolutePath());
+ return false;
+ }
+ // TODO(sky): third arg is a path, but appears to have no effect, figure out if this relates
+ // to weird caching
+ // logic mentioned above.
+ DexClassLoader bootstrapLoader = new DexClassLoader(
+ dexPath, dexOutputDir.getAbsolutePath(), null, ClassLoader.getSystemClassLoader());
try {
Class<?> loadedClass = bootstrapLoader.loadClass(BOOTSTRAP_CLASS);
Class<? extends Runnable> bootstrapClass = loadedClass.asSubclass(Runnable.class);
- Constructor<? extends Runnable> constructor = bootstrapClass.getConstructor(
- Context.class, File.class, File.class, Integer.class, Long.class);
- Runnable bootstrapRunnable = constructor.newInstance(context, bootstrap_native_library,
- application_native_library, Integer.valueOf(handle),
- Long.valueOf(runApplicationPtr));
+ Constructor<? extends Runnable> constructor =
+ bootstrapClass.getConstructor(Context.class, File.class, File.class,
+ Integer.class, Long.class, Boolean.class);
+ Runnable bootstrapRunnable = constructor.newInstance(context, bootstrapNativeLibrary,
+ applicationNativeLibrary, Integer.valueOf(handle),
+ Long.valueOf(runApplicationPtr), appType == AppType.CACHED);
bootstrapRunnable.run();
} catch (Throwable t) {
Log.e(TAG, "Running Bootstrap failed.", t);
@@ -124,6 +242,11 @@ public class AndroidHandler {
return true;
}
+ @CalledByNative
+ static String getLocalAppsDir(Context context) {
+ return ShellMain.getLocalAppsDir(context).getAbsolutePath();
+ }
+
private static File getDexOutputDir(Context context) {
return context.getDir(DEX_OUTPUT_DIRECTORY, Context.MODE_PRIVATE);
}
« no previous file with comments | « mojo/runner/android/android_handler.cc ('k') | mojo/runner/android/apk/src/org/chromium/mojo/shell/Bootstrap.java » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698