| /* |
| * Copyright (C) 2022 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.art; |
| |
| import static com.android.server.art.model.DexoptResult.DexContainerFileDexoptResult; |
| import static com.android.server.art.testing.TestingUtils.deepEq; |
| |
| import static com.google.common.truth.Truth.assertThat; |
| |
| import static org.mockito.Mockito.any; |
| import static org.mockito.Mockito.anyInt; |
| import static org.mockito.Mockito.argThat; |
| import static org.mockito.Mockito.doAnswer; |
| import static org.mockito.Mockito.doReturn; |
| import static org.mockito.Mockito.eq; |
| import static org.mockito.Mockito.inOrder; |
| import static org.mockito.Mockito.isNull; |
| import static org.mockito.Mockito.lenient; |
| import static org.mockito.Mockito.mock; |
| import static org.mockito.Mockito.never; |
| import static org.mockito.Mockito.same; |
| import static org.mockito.Mockito.times; |
| import static org.mockito.Mockito.verify; |
| import static org.mockito.Mockito.when; |
| |
| import android.os.Process; |
| import android.os.ServiceSpecificException; |
| import android.os.UserHandle; |
| |
| import androidx.test.filters.SmallTest; |
| import androidx.test.runner.AndroidJUnit4; |
| |
| import com.android.server.art.model.ArtFlags; |
| import com.android.server.art.model.DexoptParams; |
| import com.android.server.art.model.DexoptResult; |
| import com.android.server.art.testing.TestingUtils; |
| |
| import org.junit.Before; |
| import org.junit.Test; |
| import org.junit.runner.RunWith; |
| import org.mockito.InOrder; |
| |
| import java.util.ArrayList; |
| import java.util.List; |
| import java.util.concurrent.ForkJoinPool; |
| import java.util.concurrent.Future; |
| import java.util.concurrent.Semaphore; |
| import java.util.concurrent.TimeUnit; |
| import java.util.stream.Collectors; |
| |
| @SmallTest |
| @RunWith(AndroidJUnit4.class) |
| public class PrimaryDexopterTest extends PrimaryDexopterTestBase { |
| private final String mDexPath = "/data/app/foo/base.apk"; |
| private final ProfilePath mRefProfile = |
| AidlUtils.buildProfilePathForPrimaryRef(PKG_NAME, "primary"); |
| private final ProfilePath mPrebuiltProfile = AidlUtils.buildProfilePathForPrebuilt(mDexPath); |
| private final ProfilePath mDmProfile = AidlUtils.buildProfilePathForDm(mDexPath); |
| private final DexMetadataPath mDmFile = AidlUtils.buildDexMetadataPath(mDexPath); |
| private final OutputProfile mPublicOutputProfile = AidlUtils.buildOutputProfileForPrimary( |
| PKG_NAME, "primary", Process.SYSTEM_UID, SHARED_GID, true /* isOtherReadable */); |
| private final OutputProfile mPrivateOutputProfile = AidlUtils.buildOutputProfileForPrimary( |
| PKG_NAME, "primary", Process.SYSTEM_UID, SHARED_GID, false /* isOtherReadable */); |
| |
| private final String mSplit0DexPath = "/data/app/foo/split_0.apk"; |
| private final ProfilePath mSplit0RefProfile = |
| AidlUtils.buildProfilePathForPrimaryRef(PKG_NAME, "split_0.split"); |
| |
| private final int mDefaultDexoptTrigger = DexoptTrigger.COMPILER_FILTER_IS_BETTER |
| | DexoptTrigger.PRIMARY_BOOT_IMAGE_BECOMES_USABLE | DexoptTrigger.NEED_EXTRACTION; |
| private final int mBetterOrSameDexoptTrigger = DexoptTrigger.COMPILER_FILTER_IS_BETTER |
| | DexoptTrigger.COMPILER_FILTER_IS_SAME |
| | DexoptTrigger.PRIMARY_BOOT_IMAGE_BECOMES_USABLE | DexoptTrigger.NEED_EXTRACTION; |
| private final int mForceDexoptTrigger = DexoptTrigger.COMPILER_FILTER_IS_BETTER |
| | DexoptTrigger.PRIMARY_BOOT_IMAGE_BECOMES_USABLE |
| | DexoptTrigger.COMPILER_FILTER_IS_SAME | DexoptTrigger.COMPILER_FILTER_IS_WORSE |
| | DexoptTrigger.NEED_EXTRACTION; |
| |
| private final MergeProfileOptions mMergeProfileOptions = new MergeProfileOptions(); |
| |
| private final ArtdDexoptResult mArtdDexoptResult = |
| createArtdDexoptResult(false /* cancelled */); |
| |
| private DexoptParams mDexoptParams = |
| new DexoptParams.Builder("install").setCompilerFilter("speed-profile").build(); |
| |
| private PrimaryDexopter mPrimaryDexopter; |
| |
| private List<ProfilePath> mUsedProfiles; |
| private List<String> mUsedEmbeddedProfiles; |
| |
| @Before |
| public void setUp() throws Exception { |
| super.setUp(); |
| |
| // By default, none of the profiles are usable. |
| lenient().when(mArtd.isProfileUsable(any(), any())).thenReturn(false); |
| lenient() |
| .when(mArtd.copyAndRewriteProfile(any(), any(), any())) |
| .thenReturn(TestingUtils.createCopyAndRewriteProfileNoProfile()); |
| lenient() |
| .when(mArtd.copyAndRewriteEmbeddedProfile(any(), any())) |
| .thenReturn(TestingUtils.createCopyAndRewriteProfileNoProfile()); |
| |
| // By default, no DM file exists. |
| lenient().when(mArtd.getDmFileVisibility(any())).thenReturn(FileVisibility.NOT_FOUND); |
| |
| // Dexopt is by default needed and successful. |
| lenient() |
| .when(mArtd.getDexoptNeeded(any(), any(), any(), any(), anyInt())) |
| .thenReturn(dexoptIsNeeded()); |
| lenient() |
| .when(mArtd.dexopt(any(), any(), any(), any(), any(), any(), any(), any(), anyInt(), |
| any(), any())) |
| .thenReturn(mArtdDexoptResult); |
| |
| lenient() |
| .when(mArtd.createCancellationSignal()) |
| .thenReturn(mock(IArtdCancellationSignal.class)); |
| |
| mPrimaryDexopter = |
| new PrimaryDexopter(mInjector, mPkgState, mPkg, mDexoptParams, mCancellationSignal); |
| |
| mUsedProfiles = new ArrayList<>(); |
| mUsedEmbeddedProfiles = new ArrayList<>(); |
| } |
| |
| @Test |
| public void testDexoptInputVdex() throws Exception { |
| // null. |
| doReturn(dexoptIsNeeded(ArtifactsLocation.NONE_OR_ERROR)) |
| .when(mArtd) |
| .getDexoptNeeded(eq(mDexPath), eq("arm64"), any(), any(), anyInt()); |
| doReturn(mArtdDexoptResult) |
| .when(mArtd) |
| .dexopt(any(), eq(mDexPath), eq("arm64"), any(), any(), any(), isNull(), any(), |
| anyInt(), any(), any()); |
| |
| // ArtifactsPath, isInDalvikCache=true. |
| doReturn(dexoptIsNeeded(ArtifactsLocation.DALVIK_CACHE)) |
| .when(mArtd) |
| .getDexoptNeeded(eq(mDexPath), eq("arm"), any(), any(), anyInt()); |
| doReturn(mArtdDexoptResult) |
| .when(mArtd) |
| .dexopt(any(), eq(mDexPath), eq("arm"), any(), any(), any(), |
| deepEq(VdexPath.artifactsPath(AidlUtils.buildArtifactsPath( |
| mDexPath, "arm", true /* isInDalvikCache */))), |
| any(), anyInt(), any(), any()); |
| |
| // ArtifactsPath, isInDalvikCache=false. |
| doReturn(dexoptIsNeeded(ArtifactsLocation.NEXT_TO_DEX)) |
| .when(mArtd) |
| .getDexoptNeeded(eq(mSplit0DexPath), eq("arm64"), any(), any(), anyInt()); |
| doReturn(mArtdDexoptResult) |
| .when(mArtd) |
| .dexopt(any(), eq(mSplit0DexPath), eq("arm64"), any(), any(), any(), |
| deepEq(VdexPath.artifactsPath(AidlUtils.buildArtifactsPath( |
| mSplit0DexPath, "arm64", false /* isInDalvikCache */))), |
| any(), anyInt(), any(), any()); |
| |
| // DexMetadataPath. |
| doReturn(dexoptIsNeeded(ArtifactsLocation.DM)) |
| .when(mArtd) |
| .getDexoptNeeded(eq(mSplit0DexPath), eq("arm"), any(), any(), anyInt()); |
| doReturn(mArtdDexoptResult) |
| .when(mArtd) |
| .dexopt(any(), eq(mSplit0DexPath), eq("arm"), any(), any(), any(), isNull(), any(), |
| anyInt(), any(), any()); |
| |
| List<DexContainerFileDexoptResult> results = mPrimaryDexopter.dexopt(); |
| verifyStatusAllOk(results); |
| } |
| |
| @Test |
| public void testDexoptDm() throws Exception { |
| lenient() |
| .when(mArtd.getDmFileVisibility(deepEq(mDmFile))) |
| .thenReturn(FileVisibility.OTHER_READABLE); |
| |
| List<DexContainerFileDexoptResult> results = mPrimaryDexopter.dexopt(); |
| verifyStatusAllOk(results); |
| |
| verify(mArtd, times(2)) |
| .dexopt(any(), eq(mDexPath), any(), any(), any(), any(), any(), deepEq(mDmFile), |
| anyInt(), |
| argThat(dexoptOptions |
| -> dexoptOptions.compilationReason.equals("install-dm")), |
| any()); |
| verify(mArtd, times(2)) |
| .dexopt(any(), eq(mSplit0DexPath), any(), any(), any(), any(), any(), isNull(), |
| anyInt(), |
| argThat(dexoptOptions -> dexoptOptions.compilationReason.equals("install")), |
| any()); |
| } |
| |
| @Test |
| public void testDexoptUsesRefProfile() throws Exception { |
| makeProfileUsable(mRefProfile); |
| when(mArtd.getProfileVisibility(deepEq(mRefProfile))) |
| .thenReturn(FileVisibility.NOT_OTHER_READABLE); |
| |
| // Other profiles are also usable, but they shouldn't be used. |
| makeProfileUsable(mPrebuiltProfile); |
| makeProfileUsable(mDmProfile); |
| makeEmbeddedProfileUsable(mDexPath); |
| |
| List<DexContainerFileDexoptResult> results = mPrimaryDexopter.dexopt(); |
| verifyStatusAllOk(results); |
| |
| verify(mArtd).getDexoptNeeded( |
| eq(mDexPath), eq("arm64"), any(), eq("speed-profile"), eq(mDefaultDexoptTrigger)); |
| checkDexoptWithProfile( |
| verify(mArtd), mDexPath, "arm64", mRefProfile, false /* isOtherReadable */); |
| |
| verify(mArtd).getDexoptNeeded( |
| eq(mDexPath), eq("arm"), any(), eq("speed-profile"), eq(mDefaultDexoptTrigger)); |
| checkDexoptWithProfile( |
| verify(mArtd), mDexPath, "arm", mRefProfile, false /* isOtherReadable */); |
| |
| // There is no profile for split 0, so it should fall back to "verify". |
| verify(mArtd).getDexoptNeeded( |
| eq(mSplit0DexPath), eq("arm64"), any(), eq("verify"), eq(mDefaultDexoptTrigger)); |
| checkDexoptWithNoProfile(verify(mArtd), mSplit0DexPath, "arm64", "verify"); |
| |
| verify(mArtd).getDexoptNeeded( |
| eq(mSplit0DexPath), eq("arm"), any(), eq("verify"), eq(mDefaultDexoptTrigger)); |
| checkDexoptWithNoProfile(verify(mArtd), mSplit0DexPath, "arm", "verify"); |
| |
| verifyProfileNotUsed(mPrebuiltProfile); |
| verifyProfileNotUsed(mDmProfile); |
| verifyEmbeddedProfileNotUsed(mDexPath); |
| } |
| |
| @Test |
| public void testDexoptUsesPublicRefProfile() throws Exception { |
| // The ref profile is usable and public. |
| makeProfileUsable(mRefProfile); |
| when(mArtd.getProfileVisibility(deepEq(mRefProfile))) |
| .thenReturn(FileVisibility.OTHER_READABLE); |
| |
| // Other profiles are also usable, but they shouldn't be used. |
| makeProfileUsable(mPrebuiltProfile); |
| makeProfileUsable(mDmProfile); |
| makeEmbeddedProfileUsable(mDexPath); |
| |
| List<DexContainerFileDexoptResult> results = mPrimaryDexopter.dexopt(); |
| verifyStatusAllOk(results); |
| |
| checkDexoptWithProfile( |
| verify(mArtd), mDexPath, "arm64", mRefProfile, true /* isOtherReadable */); |
| checkDexoptWithProfile( |
| verify(mArtd), mDexPath, "arm", mRefProfile, true /* isOtherReadable */); |
| |
| verifyProfileNotUsed(mPrebuiltProfile); |
| verifyProfileNotUsed(mDmProfile); |
| verifyEmbeddedProfileNotUsed(mDexPath); |
| } |
| |
| @Test |
| public void testDexoptUsesPrebuiltProfile() throws Exception { |
| makeProfileUsable(mPrebuiltProfile); |
| makeProfileUsable(mDmProfile); |
| makeEmbeddedProfileUsable(mDexPath); |
| |
| List<DexContainerFileDexoptResult> results = mPrimaryDexopter.dexopt(); |
| verifyStatusAllOk(results); |
| |
| InOrder inOrder = inOrder(mArtd); |
| |
| inOrder.verify(mArtd).copyAndRewriteProfile( |
| deepEq(mPrebuiltProfile), deepEq(mPublicOutputProfile), eq(mDexPath)); |
| |
| checkDexoptWithProfile(inOrder.verify(mArtd), mDexPath, "arm64", |
| ProfilePath.tmpProfilePath(mPublicOutputProfile.profilePath), |
| true /* isOtherReadable */); |
| checkDexoptWithProfile(inOrder.verify(mArtd), mDexPath, "arm", |
| ProfilePath.tmpProfilePath(mPublicOutputProfile.profilePath), |
| true /* isOtherReadable */); |
| |
| inOrder.verify(mArtd).commitTmpProfile(deepEq(mPublicOutputProfile.profilePath)); |
| |
| verifyProfileNotUsed(mRefProfile); |
| verifyProfileNotUsed(mDmProfile); |
| verifyEmbeddedProfileNotUsed(mDexPath); |
| } |
| |
| @Test |
| public void testDexoptMergesProfiles() throws Exception { |
| setPackageInstalledForUserIds(0, 2); |
| |
| when(mArtd.mergeProfiles(any(), any(), any(), any(), any())).thenReturn(true); |
| |
| makeProfileUsable(mRefProfile); |
| when(mArtd.getProfileVisibility(deepEq(mRefProfile))) |
| .thenReturn(FileVisibility.OTHER_READABLE); |
| |
| List<DexContainerFileDexoptResult> results = mPrimaryDexopter.dexopt(); |
| verifyStatusAllOk(results); |
| |
| InOrder inOrder = inOrder(mArtd); |
| |
| inOrder.verify(mArtd).mergeProfiles( |
| deepEq(List.of(AidlUtils.buildProfilePathForPrimaryCur( |
| 0 /* userId */, PKG_NAME, "primary"), |
| AidlUtils.buildProfilePathForPrimaryCur( |
| 2 /* userId */, PKG_NAME, "primary"))), |
| deepEq(mRefProfile), deepEq(mPrivateOutputProfile), deepEq(List.of(mDexPath)), |
| deepEq(mMergeProfileOptions)); |
| |
| // It should use `mBetterOrSameDexoptTrigger` and the merged profile for both ISAs. |
| inOrder.verify(mArtd).getDexoptNeeded(eq(mDexPath), eq("arm64"), any(), eq("speed-profile"), |
| eq(mBetterOrSameDexoptTrigger)); |
| checkDexoptWithProfile(inOrder.verify(mArtd), mDexPath, "arm64", |
| ProfilePath.tmpProfilePath(mPrivateOutputProfile.profilePath), |
| false /* isOtherReadable */); |
| |
| inOrder.verify(mArtd).getDexoptNeeded(eq(mDexPath), eq("arm"), any(), eq("speed-profile"), |
| eq(mBetterOrSameDexoptTrigger)); |
| checkDexoptWithProfile(inOrder.verify(mArtd), mDexPath, "arm", |
| ProfilePath.tmpProfilePath(mPrivateOutputProfile.profilePath), |
| false /* isOtherReadable */); |
| |
| inOrder.verify(mArtd).commitTmpProfile(deepEq(mPrivateOutputProfile.profilePath)); |
| |
| inOrder.verify(mArtd).deleteProfile(deepEq( |
| AidlUtils.buildProfilePathForPrimaryCur(0 /* userId */, PKG_NAME, "primary"))); |
| inOrder.verify(mArtd).deleteProfile(deepEq( |
| AidlUtils.buildProfilePathForPrimaryCur(2 /* userId */, PKG_NAME, "primary"))); |
| } |
| |
| @Test |
| public void testDexoptMergesProfilesMergeFailed() throws Exception { |
| setPackageInstalledForUserIds(0, 2); |
| |
| when(mArtd.mergeProfiles(any(), any(), any(), any(), any())).thenReturn(false); |
| |
| makeProfileUsable(mRefProfile); |
| when(mArtd.getProfileVisibility(deepEq(mRefProfile))) |
| .thenReturn(FileVisibility.OTHER_READABLE); |
| |
| List<DexContainerFileDexoptResult> results = mPrimaryDexopter.dexopt(); |
| verifyStatusAllOk(results); |
| |
| // It should still use "speed-profile", but with the existing reference profile only. |
| verify(mArtd).getDexoptNeeded( |
| eq(mDexPath), eq("arm64"), any(), eq("speed-profile"), eq(mDefaultDexoptTrigger)); |
| checkDexoptWithProfile( |
| verify(mArtd), mDexPath, "arm64", mRefProfile, true /* isOtherReadable */); |
| |
| verify(mArtd).getDexoptNeeded( |
| eq(mDexPath), eq("arm"), any(), eq("speed-profile"), eq(mDefaultDexoptTrigger)); |
| checkDexoptWithProfile( |
| verify(mArtd), mDexPath, "arm", mRefProfile, true /* isOtherReadable */); |
| |
| verify(mArtd, never()).deleteProfile(any()); |
| verify(mArtd, never()).commitTmpProfile(any()); |
| } |
| |
| @Test |
| public void testDexoptMergesProfilesForceMerge() throws Exception { |
| mDexoptParams = mDexoptParams.toBuilder() |
| .setFlags(ArtFlags.FLAG_FORCE_MERGE_PROFILE, |
| ArtFlags.FLAG_FORCE_MERGE_PROFILE) |
| .build(); |
| mPrimaryDexopter = |
| new PrimaryDexopter(mInjector, mPkgState, mPkg, mDexoptParams, mCancellationSignal); |
| |
| setPackageInstalledForUserIds(0, 2); |
| |
| mMergeProfileOptions.forceMerge = true; |
| when(mArtd.mergeProfiles(any(), any(), any(), any(), deepEq(mMergeProfileOptions))) |
| .thenReturn(true); |
| |
| makeProfileUsable(mRefProfile); |
| when(mArtd.getProfileVisibility(deepEq(mRefProfile))) |
| .thenReturn(FileVisibility.OTHER_READABLE); |
| |
| mPrimaryDexopter.dexopt(); |
| } |
| |
| @Test |
| public void testDexoptUsesDmProfile() throws Exception { |
| makeProfileUsable(mDmProfile); |
| makeEmbeddedProfileUsable(mDexPath); |
| |
| List<DexContainerFileDexoptResult> results = mPrimaryDexopter.dexopt(); |
| verifyStatusAllOk(results); |
| |
| verify(mArtd).copyAndRewriteProfile( |
| deepEq(mDmProfile), deepEq(mPublicOutputProfile), eq(mDexPath)); |
| |
| checkDexoptWithProfile(verify(mArtd), mDexPath, "arm64", |
| ProfilePath.tmpProfilePath(mPublicOutputProfile.profilePath), |
| true /* isOtherReadable */); |
| checkDexoptWithProfile(verify(mArtd), mDexPath, "arm", |
| ProfilePath.tmpProfilePath(mPublicOutputProfile.profilePath), |
| true /* isOtherReadable */); |
| |
| verifyProfileNotUsed(mRefProfile); |
| verifyProfileNotUsed(mPrebuiltProfile); |
| verifyEmbeddedProfileNotUsed(mDexPath); |
| } |
| |
| @Test |
| public void testDexoptUsesEmbeddedProfile() throws Exception { |
| makeEmbeddedProfileUsable(mDexPath); |
| |
| List<DexContainerFileDexoptResult> results = mPrimaryDexopter.dexopt(); |
| verifyStatusAllOk(results); |
| |
| verify(mArtd).copyAndRewriteProfile( |
| deepEq(mDmProfile), deepEq(mPublicOutputProfile), eq(mDexPath)); |
| |
| checkDexoptWithProfile(verify(mArtd), mDexPath, "arm64", |
| ProfilePath.tmpProfilePath(mPublicOutputProfile.profilePath), |
| true /* isOtherReadable */); |
| checkDexoptWithProfile(verify(mArtd), mDexPath, "arm", |
| ProfilePath.tmpProfilePath(mPublicOutputProfile.profilePath), |
| true /* isOtherReadable */); |
| |
| verifyProfileNotUsed(mRefProfile); |
| verifyProfileNotUsed(mPrebuiltProfile); |
| verifyProfileNotUsed(mDmProfile); |
| } |
| |
| @Test |
| public void testDexoptExternalProfileErrors() throws Exception { |
| // Having no profile should not be reported. |
| // Having a bad profile should be reported. |
| lenient() |
| .when(mArtd.copyAndRewriteProfile(deepEq(mDmProfile), any(), any())) |
| .thenReturn(TestingUtils.createCopyAndRewriteProfileBadProfile("error_msg")); |
| |
| List<DexContainerFileDexoptResult> results = mPrimaryDexopter.dexopt(); |
| |
| assertThat(results.get(0).getStatus()).isEqualTo(DexoptResult.DEXOPT_PERFORMED); |
| assertThat(results.get(0).getExtendedStatusFlags() |
| & DexoptResult.EXTENDED_BAD_EXTERNAL_PROFILE) |
| .isNotEqualTo(0); |
| assertThat(results.get(0).getExternalProfileErrors()).containsExactly("error_msg"); |
| assertThat(results.get(1).getStatus()).isEqualTo(DexoptResult.DEXOPT_PERFORMED); |
| assertThat(results.get(1).getExtendedStatusFlags() |
| & DexoptResult.EXTENDED_BAD_EXTERNAL_PROFILE) |
| .isNotEqualTo(0); |
| assertThat(results.get(1).getExternalProfileErrors()).containsExactly("error_msg"); |
| } |
| |
| @Test |
| public void testDexoptDeletesProfileOnFailure() throws Exception { |
| makeProfileUsable(mDmProfile); |
| |
| when(mArtd.dexopt(any(), eq(mDexPath), any(), any(), any(), any(), any(), any(), anyInt(), |
| any(), any())) |
| .thenThrow(ServiceSpecificException.class); |
| |
| mPrimaryDexopter.dexopt(); |
| |
| verify(mArtd).deleteProfile( |
| deepEq(ProfilePath.tmpProfilePath(mPublicOutputProfile.profilePath))); |
| verify(mArtd, never()).commitTmpProfile(deepEq(mPublicOutputProfile.profilePath)); |
| } |
| |
| @Test |
| public void testDexoptNeedsToBeShared() throws Exception { |
| when(mDexUseManager.isPrimaryDexUsedByOtherApps(eq(PKG_NAME), eq(mDexPath))) |
| .thenReturn(true); |
| when(mDexUseManager.isPrimaryDexUsedByOtherApps(eq(PKG_NAME), eq(mSplit0DexPath))) |
| .thenReturn(true); |
| |
| // The ref profile is usable but shouldn't be used. |
| makeProfileUsable(mRefProfile); |
| |
| makeProfileUsable(mDmProfile); |
| |
| // The existing artifacts are private. |
| when(mArtd.getArtifactsVisibility( |
| argThat(artifactsPath -> artifactsPath.dexPath == mDexPath))) |
| .thenReturn(FileVisibility.NOT_OTHER_READABLE); |
| |
| List<DexContainerFileDexoptResult> results = mPrimaryDexopter.dexopt(); |
| verifyStatusAllOk(results); |
| |
| verify(mArtd).copyAndRewriteProfile( |
| deepEq(mDmProfile), deepEq(mPublicOutputProfile), eq(mDexPath)); |
| |
| // It should re-compile anyway. |
| verify(mArtd).getDexoptNeeded( |
| eq(mDexPath), eq("arm64"), any(), eq("speed-profile"), eq(mForceDexoptTrigger)); |
| checkDexoptWithProfile(verify(mArtd), mDexPath, "arm64", |
| ProfilePath.tmpProfilePath(mPublicOutputProfile.profilePath), |
| true /* isOtherReadable */); |
| |
| verify(mArtd).getDexoptNeeded( |
| eq(mDexPath), eq("arm"), any(), eq("speed-profile"), eq(mForceDexoptTrigger)); |
| checkDexoptWithProfile(verify(mArtd), mDexPath, "arm", |
| ProfilePath.tmpProfilePath(mPublicOutputProfile.profilePath), |
| true /* isOtherReadable */); |
| |
| checkDexoptWithNoProfile(verify(mArtd), mSplit0DexPath, "arm64", "speed"); |
| checkDexoptWithNoProfile(verify(mArtd), mSplit0DexPath, "arm", "speed"); |
| |
| verifyProfileNotUsed(mRefProfile); |
| verifyProfileNotUsed(mPrebuiltProfile); |
| } |
| |
| @Test |
| public void testDexoptNeedsToBeSharedArtifactsArePublic() throws Exception { |
| // Same setup as above, but the existing artifacts are public. |
| when(mDexUseManager.isPrimaryDexUsedByOtherApps(eq(PKG_NAME), eq(mDexPath))) |
| .thenReturn(true); |
| when(mDexUseManager.isPrimaryDexUsedByOtherApps(eq(PKG_NAME), eq(mSplit0DexPath))) |
| .thenReturn(true); |
| |
| makeProfileUsable(mRefProfile); |
| makeProfileUsable(mDmProfile); |
| when(mArtd.getArtifactsVisibility( |
| argThat(artifactsPath -> artifactsPath.dexPath == mDexPath))) |
| .thenReturn(FileVisibility.OTHER_READABLE); |
| |
| List<DexContainerFileDexoptResult> results = mPrimaryDexopter.dexopt(); |
| verifyStatusAllOk(results); |
| |
| // It should use the default dexopt trigger. |
| verify(mArtd).getDexoptNeeded( |
| eq(mDexPath), eq("arm64"), any(), eq("speed-profile"), eq(mDefaultDexoptTrigger)); |
| verify(mArtd).getDexoptNeeded( |
| eq(mDexPath), eq("arm"), any(), eq("speed-profile"), eq(mDefaultDexoptTrigger)); |
| } |
| |
| @Test |
| public void testDexoptUsesProfileForSplit() throws Exception { |
| makeProfileUsable(mSplit0RefProfile); |
| when(mArtd.getProfileVisibility(deepEq(mSplit0RefProfile))) |
| .thenReturn(FileVisibility.NOT_OTHER_READABLE); |
| |
| List<DexContainerFileDexoptResult> results = mPrimaryDexopter.dexopt(); |
| verifyStatusAllOk(results); |
| |
| verify(mArtd).getDexoptNeeded(eq(mSplit0DexPath), eq("arm64"), any(), eq("speed-profile"), |
| eq(mDefaultDexoptTrigger)); |
| checkDexoptWithProfile(verify(mArtd), mSplit0DexPath, "arm64", mSplit0RefProfile, |
| false /* isOtherReadable */); |
| |
| verify(mArtd).getDexoptNeeded(eq(mSplit0DexPath), eq("arm"), any(), eq("speed-profile"), |
| eq(mDefaultDexoptTrigger)); |
| checkDexoptWithProfile(verify(mArtd), mSplit0DexPath, "arm", mSplit0RefProfile, |
| false /* isOtherReadable */); |
| } |
| |
| @Test |
| public void testDexoptCancelledBeforeDexopt() throws Exception { |
| mCancellationSignal.cancel(); |
| |
| var artdCancellationSignal = mock(IArtdCancellationSignal.class); |
| when(mArtd.createCancellationSignal()).thenReturn(artdCancellationSignal); |
| |
| doAnswer(invocation -> { |
| verify(artdCancellationSignal).cancel(); |
| return createArtdDexoptResult(true /* cancelled */); |
| }) |
| .when(mArtd) |
| .dexopt(any(), any(), any(), any(), any(), any(), any(), any(), anyInt(), any(), |
| same(artdCancellationSignal)); |
| |
| // The result should only contain one element: the result of the first file with |
| // DEXOPT_CANCELLED. |
| assertThat(mPrimaryDexopter.dexopt() |
| .stream() |
| .map(DexContainerFileDexoptResult::getStatus) |
| .collect(Collectors.toList())) |
| .containsExactly(DexoptResult.DEXOPT_CANCELLED); |
| |
| // It shouldn't continue after being cancelled on the first file. |
| verify(mArtd, times(1)).createCancellationSignal(); |
| verify(mArtd, times(1)) |
| .dexopt(any(), any(), any(), any(), any(), any(), any(), any(), anyInt(), any(), |
| any()); |
| } |
| |
| @Test |
| public void testDexoptCancelledDuringDexopt() throws Exception { |
| Semaphore dexoptStarted = new Semaphore(0); |
| Semaphore dexoptCancelled = new Semaphore(0); |
| final long TIMEOUT_SEC = 10; |
| |
| var artdCancellationSignal = mock(IArtdCancellationSignal.class); |
| when(mArtd.createCancellationSignal()).thenReturn(artdCancellationSignal); |
| |
| doAnswer(invocation -> { |
| dexoptStarted.release(); |
| assertThat(dexoptCancelled.tryAcquire(TIMEOUT_SEC, TimeUnit.SECONDS)).isTrue(); |
| return createArtdDexoptResult(true /* cancelled */); |
| }) |
| .when(mArtd) |
| .dexopt(any(), any(), any(), any(), any(), any(), any(), any(), anyInt(), any(), |
| same(artdCancellationSignal)); |
| doAnswer(invocation -> { |
| dexoptCancelled.release(); |
| return null; |
| }) |
| .when(artdCancellationSignal) |
| .cancel(); |
| |
| Future<List<DexContainerFileDexoptResult>> results = |
| ForkJoinPool.commonPool().submit(() -> { return mPrimaryDexopter.dexopt(); }); |
| |
| assertThat(dexoptStarted.tryAcquire(TIMEOUT_SEC, TimeUnit.SECONDS)).isTrue(); |
| |
| mCancellationSignal.cancel(); |
| |
| assertThat(results.get() |
| .stream() |
| .map(DexContainerFileDexoptResult::getStatus) |
| .collect(Collectors.toList())) |
| .containsExactly(DexoptResult.DEXOPT_CANCELLED); |
| |
| // It shouldn't continue after being cancelled on the first file. |
| verify(mArtd, times(1)).createCancellationSignal(); |
| verify(mArtd, times(1)) |
| .dexopt(any(), any(), any(), any(), any(), any(), any(), any(), anyInt(), any(), |
| any()); |
| } |
| |
| @Test |
| public void testDexoptBaseApk() throws Exception { |
| mDexoptParams = |
| new DexoptParams.Builder("install") |
| .setCompilerFilter("speed-profile") |
| .setFlags(ArtFlags.FLAG_FOR_PRIMARY_DEX | ArtFlags.FLAG_FOR_SINGLE_SPLIT) |
| .setSplitName(null) |
| .build(); |
| mPrimaryDexopter = |
| new PrimaryDexopter(mInjector, mPkgState, mPkg, mDexoptParams, mCancellationSignal); |
| |
| List<DexContainerFileDexoptResult> results = mPrimaryDexopter.dexopt(); |
| verifyStatusAllOk(results); |
| |
| verify(mArtd, times(2)) |
| .dexopt(any(), eq(mDexPath), any(), any(), any(), any(), any(), any(), anyInt(), |
| any(), any()); |
| verify(mArtd, never()) |
| .dexopt(any(), eq(mSplit0DexPath), any(), any(), any(), any(), any(), any(), |
| anyInt(), any(), any()); |
| } |
| |
| @Test |
| public void testDexoptSplitApk() throws Exception { |
| mDexoptParams = |
| new DexoptParams.Builder("install") |
| .setCompilerFilter("speed-profile") |
| .setFlags(ArtFlags.FLAG_FOR_PRIMARY_DEX | ArtFlags.FLAG_FOR_SINGLE_SPLIT) |
| .setSplitName("split_0") |
| .build(); |
| mPrimaryDexopter = |
| new PrimaryDexopter(mInjector, mPkgState, mPkg, mDexoptParams, mCancellationSignal); |
| |
| List<DexContainerFileDexoptResult> results = mPrimaryDexopter.dexopt(); |
| verifyStatusAllOk(results); |
| |
| verify(mArtd, never()) |
| .dexopt(any(), eq(mDexPath), any(), any(), any(), any(), any(), any(), anyInt(), |
| any(), any()); |
| verify(mArtd, times(2)) |
| .dexopt(any(), eq(mSplit0DexPath), any(), any(), any(), any(), any(), any(), |
| anyInt(), any(), any()); |
| } |
| |
| @Test |
| public void testDexoptStorageLow() throws Exception { |
| when(mStorageManager.getAllocatableBytes(any())).thenReturn(1l, 0l, 0l, 1l); |
| |
| mDexoptParams = |
| new DexoptParams.Builder("install") |
| .setCompilerFilter("speed-profile") |
| .setFlags(ArtFlags.FLAG_FOR_PRIMARY_DEX | ArtFlags.FLAG_SKIP_IF_STORAGE_LOW) |
| .build(); |
| mPrimaryDexopter = |
| new PrimaryDexopter(mInjector, mPkgState, mPkg, mDexoptParams, mCancellationSignal); |
| |
| List<DexContainerFileDexoptResult> results = mPrimaryDexopter.dexopt(); |
| assertThat(results.get(0).getStatus()).isEqualTo(DexoptResult.DEXOPT_PERFORMED); |
| assertThat( |
| results.get(0).getExtendedStatusFlags() & DexoptResult.EXTENDED_SKIPPED_STORAGE_LOW) |
| .isEqualTo(0); |
| assertThat(results.get(1).getStatus()).isEqualTo(DexoptResult.DEXOPT_SKIPPED); |
| assertThat( |
| results.get(1).getExtendedStatusFlags() & DexoptResult.EXTENDED_SKIPPED_STORAGE_LOW) |
| .isNotEqualTo(0); |
| assertThat(results.get(2).getStatus()).isEqualTo(DexoptResult.DEXOPT_SKIPPED); |
| assertThat( |
| results.get(2).getExtendedStatusFlags() & DexoptResult.EXTENDED_SKIPPED_STORAGE_LOW) |
| .isNotEqualTo(0); |
| assertThat(results.get(3).getStatus()).isEqualTo(DexoptResult.DEXOPT_PERFORMED); |
| assertThat( |
| results.get(3).getExtendedStatusFlags() & DexoptResult.EXTENDED_SKIPPED_STORAGE_LOW) |
| .isEqualTo(0); |
| |
| verify(mArtd, times(2)) |
| .dexopt(any(), any(), any(), any(), any(), any(), any(), any(), anyInt(), any(), |
| any()); |
| } |
| |
| @Test |
| public void testDexoptDexStatus() throws Exception { |
| lenient() |
| .when(mArtd.getDexoptNeeded(any(), any(), any(), any(), anyInt())) |
| .thenReturn(dexoptIsNotNeeded(false /* hasDexCode */), |
| dexoptIsNotNeeded(false /* hasDexCode */), |
| dexoptIsNotNeeded(true /* hasDexCode */), dexoptIsNeeded()); |
| |
| mDexoptParams = new DexoptParams.Builder("install") |
| .setCompilerFilter("speed-profile") |
| .setFlags(ArtFlags.FLAG_FOR_PRIMARY_DEX) |
| .build(); |
| mPrimaryDexopter = |
| new PrimaryDexopter(mInjector, mPkgState, mPkg, mDexoptParams, mCancellationSignal); |
| |
| List<DexContainerFileDexoptResult> results = mPrimaryDexopter.dexopt(); |
| assertThat(results.get(0).getStatus()).isEqualTo(DexoptResult.DEXOPT_SKIPPED); |
| assertThat( |
| results.get(0).getExtendedStatusFlags() & DexoptResult.EXTENDED_SKIPPED_NO_DEX_CODE) |
| .isNotEqualTo(0); |
| assertThat(results.get(1).getStatus()).isEqualTo(DexoptResult.DEXOPT_SKIPPED); |
| assertThat( |
| results.get(1).getExtendedStatusFlags() & DexoptResult.EXTENDED_SKIPPED_NO_DEX_CODE) |
| .isNotEqualTo(0); |
| assertThat(results.get(2).getStatus()).isEqualTo(DexoptResult.DEXOPT_SKIPPED); |
| assertThat( |
| results.get(2).getExtendedStatusFlags() & DexoptResult.EXTENDED_SKIPPED_NO_DEX_CODE) |
| .isEqualTo(0); |
| assertThat(results.get(3).getStatus()).isEqualTo(DexoptResult.DEXOPT_PERFORMED); |
| assertThat( |
| results.get(3).getExtendedStatusFlags() & DexoptResult.EXTENDED_SKIPPED_NO_DEX_CODE) |
| .isEqualTo(0); |
| } |
| |
| private void checkDexoptWithProfile(IArtd artd, String dexPath, String isa, ProfilePath profile, |
| boolean isOtherReadable) throws Exception { |
| artd.dexopt(argThat(artifacts |
| -> artifacts.permissionSettings.fileFsPermission.isOtherReadable |
| == isOtherReadable), |
| eq(dexPath), eq(isa), any(), eq("speed-profile"), deepEq(profile), any(), any(), |
| anyInt(), argThat(dexoptOptions -> dexoptOptions.generateAppImage == true), any()); |
| } |
| |
| private void checkDexoptWithNoProfile( |
| IArtd artd, String dexPath, String isa, String compilerFilter) throws Exception { |
| artd.dexopt( |
| argThat(artifacts |
| -> artifacts.permissionSettings.fileFsPermission.isOtherReadable == true), |
| eq(dexPath), eq(isa), any(), eq(compilerFilter), isNull(), any(), any(), anyInt(), |
| argThat(dexoptOptions -> dexoptOptions.generateAppImage == false), any()); |
| } |
| |
| private void verifyProfileNotUsed(ProfilePath profile) throws Exception { |
| assertThat(mUsedProfiles) |
| .comparingElementsUsing(TestingUtils.<ProfilePath>deepEquality()) |
| .doesNotContain(profile); |
| } |
| |
| private void verifyEmbeddedProfileNotUsed(String dexPath) throws Exception { |
| assertThat(mUsedEmbeddedProfiles).doesNotContain(dexPath); |
| } |
| |
| private void makeProfileUsable(ProfilePath profile) throws Exception { |
| lenient().when(mArtd.isProfileUsable(deepEq(profile), any())).thenAnswer(invocation -> { |
| mUsedProfiles.add(invocation.<ProfilePath>getArgument(0)); |
| return true; |
| }); |
| lenient() |
| .when(mArtd.copyAndRewriteProfile(deepEq(profile), any(), any())) |
| .thenAnswer(invocation -> { |
| mUsedProfiles.add(invocation.<ProfilePath>getArgument(0)); |
| return TestingUtils.createCopyAndRewriteProfileSuccess(); |
| }); |
| } |
| |
| private void makeEmbeddedProfileUsable(String dexPath) throws Exception { |
| lenient() |
| .when(mArtd.copyAndRewriteEmbeddedProfile(any(), eq(dexPath))) |
| .thenAnswer(invocation -> { |
| mUsedEmbeddedProfiles.add(invocation.<String>getArgument(1)); |
| return TestingUtils.createCopyAndRewriteProfileSuccess(); |
| }); |
| } |
| |
| private void verifyStatusAllOk(List<DexContainerFileDexoptResult> results) { |
| for (DexContainerFileDexoptResult result : results) { |
| assertThat(result.getStatus()).isEqualTo(DexoptResult.DEXOPT_PERFORMED); |
| assertThat(result.getExtendedStatusFlags()).isEqualTo(0); |
| assertThat(result.getExternalProfileErrors()).isEmpty(); |
| } |
| } |
| |
| /** Dexopter relies on this information to determine which current profiles to check. */ |
| private void setPackageInstalledForUserIds(int... userIds) { |
| for (int userId : userIds) { |
| when(mPkgState.getStateForUser(eq(UserHandle.of(userId)))) |
| .thenReturn(mPkgUserStateInstalled); |
| } |
| } |
| } |