2019-05-23 00:50:08 -07:00
|
|
|
/*
|
|
|
|
|
* Copyright (C) 2019 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.launcher3.testing;
|
|
|
|
|
|
2019-08-14 17:45:45 -07:00
|
|
|
import static android.graphics.Bitmap.Config.ARGB_8888;
|
2019-08-15 14:53:41 -07:00
|
|
|
|
2019-09-20 09:37:11 -07:00
|
|
|
import static com.android.launcher3.util.Executors.MAIN_EXECUTOR;
|
|
|
|
|
|
2019-05-23 00:50:08 -07:00
|
|
|
import android.content.Context;
|
2019-08-14 17:45:45 -07:00
|
|
|
import android.graphics.Bitmap;
|
|
|
|
|
import android.graphics.Color;
|
2019-05-23 00:50:08 -07:00
|
|
|
import android.os.Bundle;
|
2019-08-14 17:45:45 -07:00
|
|
|
import android.os.Debug;
|
2020-01-08 18:48:24 +00:00
|
|
|
import android.system.Os;
|
2019-09-19 18:26:47 -07:00
|
|
|
import android.view.View;
|
2019-05-23 00:50:08 -07:00
|
|
|
|
2019-10-25 18:12:25 -07:00
|
|
|
import androidx.annotation.Keep;
|
|
|
|
|
|
2019-05-23 00:50:08 -07:00
|
|
|
import com.android.launcher3.DeviceProfile;
|
|
|
|
|
import com.android.launcher3.InvariantDeviceProfile;
|
|
|
|
|
import com.android.launcher3.Launcher;
|
|
|
|
|
import com.android.launcher3.LauncherAppState;
|
|
|
|
|
import com.android.launcher3.LauncherState;
|
|
|
|
|
import com.android.launcher3.R;
|
2019-06-04 13:59:43 -07:00
|
|
|
import com.android.launcher3.allapps.AllAppsStore;
|
2019-05-23 00:50:08 -07:00
|
|
|
import com.android.launcher3.util.ResourceBasedOverride;
|
|
|
|
|
|
2019-08-14 17:45:45 -07:00
|
|
|
import java.util.LinkedList;
|
2019-10-04 11:00:01 -07:00
|
|
|
import java.util.concurrent.CountDownLatch;
|
2019-06-04 13:59:43 -07:00
|
|
|
import java.util.concurrent.ExecutionException;
|
2019-10-04 11:00:01 -07:00
|
|
|
import java.util.concurrent.TimeUnit;
|
2019-06-04 13:59:43 -07:00
|
|
|
|
2019-05-23 00:50:08 -07:00
|
|
|
public class TestInformationHandler implements ResourceBasedOverride {
|
|
|
|
|
|
|
|
|
|
public static TestInformationHandler newInstance(Context context) {
|
|
|
|
|
return Overrides.getObject(TestInformationHandler.class,
|
|
|
|
|
context, R.string.test_information_handler_class);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected Context mContext;
|
|
|
|
|
protected DeviceProfile mDeviceProfile;
|
|
|
|
|
protected LauncherAppState mLauncherAppState;
|
|
|
|
|
protected Launcher mLauncher;
|
2019-08-14 17:45:45 -07:00
|
|
|
private static LinkedList mLeaks;
|
2019-05-23 00:50:08 -07:00
|
|
|
|
|
|
|
|
public void init(Context context) {
|
|
|
|
|
mContext = context;
|
|
|
|
|
mDeviceProfile = InvariantDeviceProfile.INSTANCE.
|
|
|
|
|
get(context).getDeviceProfile(context);
|
|
|
|
|
mLauncherAppState = LauncherAppState.getInstanceNoCreate();
|
2019-09-26 17:05:31 -07:00
|
|
|
mLauncher = Launcher.ACTIVITY_TRACKER.getCreatedActivity();
|
2019-05-23 00:50:08 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Bundle call(String method) {
|
|
|
|
|
final Bundle response = new Bundle();
|
|
|
|
|
switch (method) {
|
|
|
|
|
case TestProtocol.REQUEST_ALL_APPS_TO_OVERVIEW_SWIPE_HEIGHT: {
|
|
|
|
|
if (mLauncher == null) return null;
|
|
|
|
|
|
|
|
|
|
final float progress = LauncherState.OVERVIEW.getVerticalProgress(mLauncher)
|
|
|
|
|
- LauncherState.ALL_APPS.getVerticalProgress(mLauncher);
|
|
|
|
|
final float distance = mLauncher.getAllAppsController().getShiftRange() * progress;
|
|
|
|
|
response.putInt(TestProtocol.TEST_INFO_RESPONSE_FIELD, (int) distance);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case TestProtocol.REQUEST_HOME_TO_ALL_APPS_SWIPE_HEIGHT: {
|
|
|
|
|
if (mLauncher == null) return null;
|
|
|
|
|
|
|
|
|
|
final float progress = LauncherState.NORMAL.getVerticalProgress(mLauncher)
|
|
|
|
|
- LauncherState.ALL_APPS.getVerticalProgress(mLauncher);
|
|
|
|
|
final float distance = mLauncher.getAllAppsController().getShiftRange() * progress;
|
|
|
|
|
response.putInt(TestProtocol.TEST_INFO_RESPONSE_FIELD, (int) distance);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2019-06-28 12:04:36 -07:00
|
|
|
case TestProtocol.REQUEST_IS_LAUNCHER_INITIALIZED: {
|
2019-09-27 18:44:04 -07:00
|
|
|
response.putBoolean(TestProtocol.TEST_INFO_RESPONSE_FIELD, isLauncherInitialized());
|
2019-06-28 12:04:36 -07:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2019-05-23 00:50:08 -07:00
|
|
|
case TestProtocol.REQUEST_ENABLE_DEBUG_TRACING:
|
|
|
|
|
TestProtocol.sDebugTracing = true;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case TestProtocol.REQUEST_DISABLE_DEBUG_TRACING:
|
|
|
|
|
TestProtocol.sDebugTracing = false;
|
|
|
|
|
break;
|
2019-06-04 13:59:43 -07:00
|
|
|
|
|
|
|
|
case TestProtocol.REQUEST_FREEZE_APP_LIST:
|
2019-08-15 14:53:41 -07:00
|
|
|
MAIN_EXECUTOR.execute(() ->
|
2019-06-04 13:59:43 -07:00
|
|
|
mLauncher.getAppsView().getAppsStore().enableDeferUpdates(
|
|
|
|
|
AllAppsStore.DEFER_UPDATES_TEST));
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case TestProtocol.REQUEST_UNFREEZE_APP_LIST:
|
2019-08-15 14:53:41 -07:00
|
|
|
MAIN_EXECUTOR.execute(() ->
|
2019-06-04 13:59:43 -07:00
|
|
|
mLauncher.getAppsView().getAppsStore().disableDeferUpdates(
|
|
|
|
|
AllAppsStore.DEFER_UPDATES_TEST));
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case TestProtocol.REQUEST_APP_LIST_FREEZE_FLAGS: {
|
|
|
|
|
try {
|
2019-08-15 14:53:41 -07:00
|
|
|
final int deferUpdatesFlags = MAIN_EXECUTOR.submit(() ->
|
2019-06-04 13:59:43 -07:00
|
|
|
mLauncher.getAppsView().getAppsStore().getDeferUpdatesFlags()).get();
|
|
|
|
|
response.putInt(TestProtocol.TEST_INFO_RESPONSE_FIELD,
|
|
|
|
|
deferUpdatesFlags);
|
2019-10-11 12:58:18 -07:00
|
|
|
} catch (ExecutionException | InterruptedException e) {
|
2019-06-04 13:59:43 -07:00
|
|
|
throw new RuntimeException(e);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
2019-08-08 11:49:24 -07:00
|
|
|
|
2019-09-20 09:37:11 -07:00
|
|
|
case TestProtocol.REQUEST_APPS_LIST_SCROLL_Y: {
|
|
|
|
|
try {
|
|
|
|
|
final int deferUpdatesFlags = MAIN_EXECUTOR.submit(() ->
|
|
|
|
|
mLauncher.getAppsView().getActiveRecyclerView().getCurrentScrollY())
|
|
|
|
|
.get();
|
|
|
|
|
response.putInt(TestProtocol.TEST_INFO_RESPONSE_FIELD,
|
|
|
|
|
deferUpdatesFlags);
|
2019-10-11 12:58:18 -07:00
|
|
|
} catch (ExecutionException | InterruptedException e) {
|
2019-09-20 09:37:11 -07:00
|
|
|
throw new RuntimeException(e);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2020-01-08 18:48:24 +00:00
|
|
|
case TestProtocol.REQUEST_PID: {
|
|
|
|
|
response.putInt(TestProtocol.TEST_INFO_RESPONSE_FIELD, Os.getpid());
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2019-08-14 17:45:45 -07:00
|
|
|
case TestProtocol.REQUEST_TOTAL_PSS_KB: {
|
2019-10-04 11:00:01 -07:00
|
|
|
runGcAndFinalizersSync();
|
2019-08-14 17:45:45 -07:00
|
|
|
Debug.MemoryInfo mem = new Debug.MemoryInfo();
|
|
|
|
|
Debug.getMemoryInfo(mem);
|
|
|
|
|
response.putInt(TestProtocol.TEST_INFO_RESPONSE_FIELD, mem.getTotalPss());
|
2019-08-08 11:49:24 -07:00
|
|
|
break;
|
|
|
|
|
}
|
2019-08-14 17:45:45 -07:00
|
|
|
|
|
|
|
|
case TestProtocol.REQUEST_JAVA_LEAK: {
|
|
|
|
|
if (mLeaks == null) mLeaks = new LinkedList();
|
|
|
|
|
|
|
|
|
|
// Allocate and dirty the memory.
|
|
|
|
|
final int leakSize = 1024 * 1024;
|
|
|
|
|
final byte[] bytes = new byte[leakSize];
|
|
|
|
|
for (int i = 0; i < leakSize; i += 239) {
|
|
|
|
|
bytes[i] = (byte) (i % 256);
|
|
|
|
|
}
|
|
|
|
|
mLeaks.add(bytes);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case TestProtocol.REQUEST_NATIVE_LEAK: {
|
|
|
|
|
if (mLeaks == null) mLeaks = new LinkedList();
|
|
|
|
|
|
|
|
|
|
// Allocate and dirty a bitmap.
|
|
|
|
|
final Bitmap bitmap = Bitmap.createBitmap(512, 512, ARGB_8888);
|
|
|
|
|
bitmap.eraseColor(Color.RED);
|
|
|
|
|
mLeaks.add(bitmap);
|
|
|
|
|
break;
|
|
|
|
|
}
|
2019-09-19 18:26:47 -07:00
|
|
|
|
|
|
|
|
case TestProtocol.REQUEST_VIEW_LEAK: {
|
|
|
|
|
if (mLeaks == null) mLeaks = new LinkedList();
|
|
|
|
|
|
|
|
|
|
mLeaks.add(new View(mContext));
|
|
|
|
|
break;
|
|
|
|
|
}
|
2019-11-12 16:57:48 -08:00
|
|
|
|
|
|
|
|
case TestProtocol.REQUEST_ICON_HEIGHT: {
|
|
|
|
|
response.putInt(TestProtocol.TEST_INFO_RESPONSE_FIELD,
|
|
|
|
|
mDeviceProfile.allAppsCellHeightPx);
|
|
|
|
|
break;
|
|
|
|
|
}
|
2019-05-23 00:50:08 -07:00
|
|
|
}
|
|
|
|
|
return response;
|
|
|
|
|
}
|
2019-09-27 18:44:04 -07:00
|
|
|
|
|
|
|
|
protected boolean isLauncherInitialized() {
|
2019-10-02 14:27:25 -07:00
|
|
|
return Launcher.ACTIVITY_TRACKER.getCreatedActivity() == null
|
|
|
|
|
|| LauncherAppState.getInstance(mContext).getModel().isModelLoaded();
|
2019-09-27 18:44:04 -07:00
|
|
|
}
|
2019-10-04 11:00:01 -07:00
|
|
|
|
|
|
|
|
private static void runGcAndFinalizersSync() {
|
|
|
|
|
Runtime.getRuntime().gc();
|
|
|
|
|
Runtime.getRuntime().runFinalization();
|
|
|
|
|
|
|
|
|
|
final CountDownLatch fence = new CountDownLatch(1);
|
2019-10-25 18:12:25 -07:00
|
|
|
createFinalizationObserver(fence);
|
|
|
|
|
try {
|
|
|
|
|
do {
|
|
|
|
|
Runtime.getRuntime().gc();
|
|
|
|
|
Runtime.getRuntime().runFinalization();
|
|
|
|
|
} while (!fence.await(100, TimeUnit.MILLISECONDS));
|
|
|
|
|
} catch (InterruptedException ex) {
|
|
|
|
|
throw new RuntimeException(ex);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Create the observer in the scope of a method to minimize the chance that
|
|
|
|
|
// it remains live in a DEX/machine register at the point of the fence guard.
|
|
|
|
|
// This must be kept to avoid R8 inlining it.
|
|
|
|
|
@Keep
|
|
|
|
|
private static void createFinalizationObserver(CountDownLatch fence) {
|
2019-10-04 11:00:01 -07:00
|
|
|
new Object() {
|
|
|
|
|
@Override
|
|
|
|
|
protected void finalize() throws Throwable {
|
|
|
|
|
try {
|
|
|
|
|
fence.countDown();
|
|
|
|
|
} finally {
|
|
|
|
|
super.finalize();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
}
|
2019-09-17 13:03:18 -07:00
|
|
|
}
|