Loading services/core/java/com/android/server/pm/OtaDexoptService.java +2 −1 Original line number Diff line number Diff line Loading @@ -314,7 +314,8 @@ public class OtaDexoptService extends IOtaDexopt.Stub { optimizer.performDexOpt(pkg, libraryDependencies, null /* ISAs */, false /* checkProfiles */, getCompilerFilterForReason(compilationReason), null /* CompilerStats.PackageStats */); null /* CompilerStats.PackageStats */, mPackageManagerService.getDexManager().isUsedByOtherApps(pkg.packageName)); return commands; } Loading services/core/java/com/android/server/pm/PackageDexOptimizer.java +5 −38 Original line number Diff line number Diff line Loading @@ -104,7 +104,7 @@ public class PackageDexOptimizer { */ int performDexOpt(PackageParser.Package pkg, String[] sharedLibraries, String[] instructionSets, boolean checkProfiles, String targetCompilationFilter, CompilerStats.PackageStats packageStats) { CompilerStats.PackageStats packageStats, boolean isUsedByOtherApps) { if (!canOptimizePackage(pkg)) { return DEX_OPT_SKIPPED; } Loading @@ -119,7 +119,7 @@ public class PackageDexOptimizer { } try { return performDexOptLI(pkg, sharedLibraries, instructionSets, checkProfiles, targetCompilationFilter, packageStats); targetCompilationFilter, packageStats, isUsedByOtherApps); } finally { if (useLock) { mDexoptWakeLock.release(); Loading @@ -135,7 +135,8 @@ public class PackageDexOptimizer { @GuardedBy("mInstallLock") private int performDexOptLI(PackageParser.Package pkg, String[] sharedLibraries, String[] targetInstructionSets, boolean checkForProfileUpdates, String targetCompilerFilter, CompilerStats.PackageStats packageStats) { String targetCompilerFilter, CompilerStats.PackageStats packageStats, boolean isUsedByOtherApps) { final String[] instructionSets = targetInstructionSets != null ? targetInstructionSets : getAppDexInstructionSets(pkg.applicationInfo); final String[] dexCodeInstructionSets = getDexCodeInstructionSets(instructionSets); Loading @@ -143,7 +144,7 @@ public class PackageDexOptimizer { final int sharedGid = UserHandle.getSharedAppGid(pkg.applicationInfo.uid); final String compilerFilter = getRealCompilerFilter(pkg.applicationInfo, targetCompilerFilter, isUsedByOtherApps(pkg)); targetCompilerFilter, isUsedByOtherApps); final boolean profileUpdated = checkForProfileUpdates && isProfileUpdated(pkg, sharedGid, compilerFilter); Loading Loading @@ -477,40 +478,6 @@ public class PackageDexOptimizer { mSystemReady = true; } /** * Returns true if the profiling data collected for the given app indicate * that the apps's APK has been loaded by another app. * Note that this returns false for all forward-locked apps and apps without * any collected profiling data. */ public static boolean isUsedByOtherApps(PackageParser.Package pkg) { if (pkg.isForwardLocked()) { // Skip the check for forward locked packages since they don't share their code. return false; } for (String apkPath : pkg.getAllCodePathsExcludingResourceOnly()) { try { apkPath = PackageManagerServiceUtils.realpath(new File(apkPath)); } catch (IOException e) { // Log an error but continue without it. Slog.w(TAG, "Failed to get canonical path", e); continue; } String useMarker = apkPath.replace('/', '@'); final int[] currentUserIds = UserManagerService.getInstance().getUserIds(); for (int i = 0; i < currentUserIds.length; i++) { File profileDir = Environment.getDataProfilesDeForeignDexDirectory(currentUserIds[i]); File foreignUseMark = new File(profileDir, useMarker); if (foreignUseMark.exists()) { return true; } } } return false; } private String printDexoptFlags(int flags) { ArrayList<String> flagsList = new ArrayList<>(); Loading services/core/java/com/android/server/pm/PackageManagerService.java +6 −3 Original line number Diff line number Diff line Loading @@ -8441,11 +8441,13 @@ public class PackageManagerService extends IPackageManager.Stub { pdo.performDexOpt(depPackage, null /* sharedLibraries */, instructionSets, false /* checkProfiles */, getCompilerFilterForReason(REASON_NON_SYSTEM_LIBRARY), getOrCreateCompilerPackageStats(depPackage)); getOrCreateCompilerPackageStats(depPackage), mDexManager.isUsedByOtherApps(p.packageName)); } } return pdo.performDexOpt(p, p.usesLibraryFiles, instructionSets, checkProfiles, targetCompilerFilter, getOrCreateCompilerPackageStats(p)); targetCompilerFilter, getOrCreateCompilerPackageStats(p), mDexManager.isUsedByOtherApps(p.packageName)); } // Performs dexopt on the used secondary dex files belonging to the given package. Loading Loading @@ -16965,7 +16967,8 @@ public class PackageManagerService extends IPackageManager.Stub { mPackageDexOptimizer.performDexOpt(pkg, pkg.usesLibraryFiles, null /* instructionSets */, false /* checkProfiles */, getCompilerFilterForReason(REASON_INSTALL), getOrCreateCompilerPackageStats(pkg)); getOrCreateCompilerPackageStats(pkg), mDexManager.isUsedByOtherApps(pkg.packageName)); Trace.traceEnd(TRACE_TAG_PACKAGE_MANAGER); // Notify BackgroundDexOptJobService that the package has been changed. services/core/java/com/android/server/pm/PackageManagerServiceUtils.java +2 −1 Original line number Diff line number Diff line Loading @@ -133,7 +133,8 @@ public class PackageManagerServiceUtils { sortTemp, packageManagerService); // Give priority to apps used by other apps. applyPackageFilter((pkg) -> PackageDexOptimizer.isUsedByOtherApps(pkg), result, applyPackageFilter((pkg) -> packageManagerService.getDexManager().isUsedByOtherApps(pkg.packageName), result, remainingPkgs, sortTemp, packageManagerService); // Filter out packages that aren't recently used, add all remaining apps. Loading services/core/java/com/android/server/pm/dex/DexManager.java +17 −0 Original line number Diff line number Diff line Loading @@ -358,6 +358,23 @@ public class DexManager { return mPackageDexUsage.getAllPackagesWithSecondaryDexFiles(); } /** * Return true if the profiling data collected for the given app indicate * that the apps's APK has been loaded by another app. * Note that this returns false for all apps without any collected profiling data. */ public boolean isUsedByOtherApps(String packageName) { PackageUseInfo useInfo = getPackageUseInfo(packageName); if (useInfo == null) { // No use info, means the package was not used or it was used but not by other apps. // Note that right now we might prune packages which are not used by other apps. // TODO(calin): maybe we should not (prune) so we can have an accurate view when we try // to access the package use. return false; } return useInfo.isUsedByOtherApps(); } /** * Retrieves the package which owns the given dexPath. */ Loading Loading
services/core/java/com/android/server/pm/OtaDexoptService.java +2 −1 Original line number Diff line number Diff line Loading @@ -314,7 +314,8 @@ public class OtaDexoptService extends IOtaDexopt.Stub { optimizer.performDexOpt(pkg, libraryDependencies, null /* ISAs */, false /* checkProfiles */, getCompilerFilterForReason(compilationReason), null /* CompilerStats.PackageStats */); null /* CompilerStats.PackageStats */, mPackageManagerService.getDexManager().isUsedByOtherApps(pkg.packageName)); return commands; } Loading
services/core/java/com/android/server/pm/PackageDexOptimizer.java +5 −38 Original line number Diff line number Diff line Loading @@ -104,7 +104,7 @@ public class PackageDexOptimizer { */ int performDexOpt(PackageParser.Package pkg, String[] sharedLibraries, String[] instructionSets, boolean checkProfiles, String targetCompilationFilter, CompilerStats.PackageStats packageStats) { CompilerStats.PackageStats packageStats, boolean isUsedByOtherApps) { if (!canOptimizePackage(pkg)) { return DEX_OPT_SKIPPED; } Loading @@ -119,7 +119,7 @@ public class PackageDexOptimizer { } try { return performDexOptLI(pkg, sharedLibraries, instructionSets, checkProfiles, targetCompilationFilter, packageStats); targetCompilationFilter, packageStats, isUsedByOtherApps); } finally { if (useLock) { mDexoptWakeLock.release(); Loading @@ -135,7 +135,8 @@ public class PackageDexOptimizer { @GuardedBy("mInstallLock") private int performDexOptLI(PackageParser.Package pkg, String[] sharedLibraries, String[] targetInstructionSets, boolean checkForProfileUpdates, String targetCompilerFilter, CompilerStats.PackageStats packageStats) { String targetCompilerFilter, CompilerStats.PackageStats packageStats, boolean isUsedByOtherApps) { final String[] instructionSets = targetInstructionSets != null ? targetInstructionSets : getAppDexInstructionSets(pkg.applicationInfo); final String[] dexCodeInstructionSets = getDexCodeInstructionSets(instructionSets); Loading @@ -143,7 +144,7 @@ public class PackageDexOptimizer { final int sharedGid = UserHandle.getSharedAppGid(pkg.applicationInfo.uid); final String compilerFilter = getRealCompilerFilter(pkg.applicationInfo, targetCompilerFilter, isUsedByOtherApps(pkg)); targetCompilerFilter, isUsedByOtherApps); final boolean profileUpdated = checkForProfileUpdates && isProfileUpdated(pkg, sharedGid, compilerFilter); Loading Loading @@ -477,40 +478,6 @@ public class PackageDexOptimizer { mSystemReady = true; } /** * Returns true if the profiling data collected for the given app indicate * that the apps's APK has been loaded by another app. * Note that this returns false for all forward-locked apps and apps without * any collected profiling data. */ public static boolean isUsedByOtherApps(PackageParser.Package pkg) { if (pkg.isForwardLocked()) { // Skip the check for forward locked packages since they don't share their code. return false; } for (String apkPath : pkg.getAllCodePathsExcludingResourceOnly()) { try { apkPath = PackageManagerServiceUtils.realpath(new File(apkPath)); } catch (IOException e) { // Log an error but continue without it. Slog.w(TAG, "Failed to get canonical path", e); continue; } String useMarker = apkPath.replace('/', '@'); final int[] currentUserIds = UserManagerService.getInstance().getUserIds(); for (int i = 0; i < currentUserIds.length; i++) { File profileDir = Environment.getDataProfilesDeForeignDexDirectory(currentUserIds[i]); File foreignUseMark = new File(profileDir, useMarker); if (foreignUseMark.exists()) { return true; } } } return false; } private String printDexoptFlags(int flags) { ArrayList<String> flagsList = new ArrayList<>(); Loading
services/core/java/com/android/server/pm/PackageManagerService.java +6 −3 Original line number Diff line number Diff line Loading @@ -8441,11 +8441,13 @@ public class PackageManagerService extends IPackageManager.Stub { pdo.performDexOpt(depPackage, null /* sharedLibraries */, instructionSets, false /* checkProfiles */, getCompilerFilterForReason(REASON_NON_SYSTEM_LIBRARY), getOrCreateCompilerPackageStats(depPackage)); getOrCreateCompilerPackageStats(depPackage), mDexManager.isUsedByOtherApps(p.packageName)); } } return pdo.performDexOpt(p, p.usesLibraryFiles, instructionSets, checkProfiles, targetCompilerFilter, getOrCreateCompilerPackageStats(p)); targetCompilerFilter, getOrCreateCompilerPackageStats(p), mDexManager.isUsedByOtherApps(p.packageName)); } // Performs dexopt on the used secondary dex files belonging to the given package. Loading Loading @@ -16965,7 +16967,8 @@ public class PackageManagerService extends IPackageManager.Stub { mPackageDexOptimizer.performDexOpt(pkg, pkg.usesLibraryFiles, null /* instructionSets */, false /* checkProfiles */, getCompilerFilterForReason(REASON_INSTALL), getOrCreateCompilerPackageStats(pkg)); getOrCreateCompilerPackageStats(pkg), mDexManager.isUsedByOtherApps(pkg.packageName)); Trace.traceEnd(TRACE_TAG_PACKAGE_MANAGER); // Notify BackgroundDexOptJobService that the package has been changed.
services/core/java/com/android/server/pm/PackageManagerServiceUtils.java +2 −1 Original line number Diff line number Diff line Loading @@ -133,7 +133,8 @@ public class PackageManagerServiceUtils { sortTemp, packageManagerService); // Give priority to apps used by other apps. applyPackageFilter((pkg) -> PackageDexOptimizer.isUsedByOtherApps(pkg), result, applyPackageFilter((pkg) -> packageManagerService.getDexManager().isUsedByOtherApps(pkg.packageName), result, remainingPkgs, sortTemp, packageManagerService); // Filter out packages that aren't recently used, add all remaining apps. Loading
services/core/java/com/android/server/pm/dex/DexManager.java +17 −0 Original line number Diff line number Diff line Loading @@ -358,6 +358,23 @@ public class DexManager { return mPackageDexUsage.getAllPackagesWithSecondaryDexFiles(); } /** * Return true if the profiling data collected for the given app indicate * that the apps's APK has been loaded by another app. * Note that this returns false for all apps without any collected profiling data. */ public boolean isUsedByOtherApps(String packageName) { PackageUseInfo useInfo = getPackageUseInfo(packageName); if (useInfo == null) { // No use info, means the package was not used or it was used but not by other apps. // Note that right now we might prune packages which are not used by other apps. // TODO(calin): maybe we should not (prune) so we can have an accurate view when we try // to access the package use. return false; } return useInfo.isUsedByOtherApps(); } /** * Retrieves the package which owns the given dexPath. */ Loading