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
|
|
|
|
2020-02-07 10:47:03 -08:00
|
|
|
import static com.android.launcher3.allapps.AllAppsStore.DEFER_UPDATES_TEST;
|
2019-09-20 09:37:11 -07:00
|
|
|
import static com.android.launcher3.util.Executors.MAIN_EXECUTOR;
|
|
|
|
|
|
2020-02-07 10:47:03 -08:00
|
|
|
import android.annotation.TargetApi;
|
|
|
|
|
import android.app.Activity;
|
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;
|
2020-02-07 10:47:03 -08:00
|
|
|
import android.graphics.Insets;
|
|
|
|
|
import android.os.Build;
|
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;
|
2020-01-27 17:44:44 -08:00
|
|
|
import android.util.Log;
|
2019-09-19 18:26:47 -07:00
|
|
|
import android.view.View;
|
2020-02-07 10:47:03 -08:00
|
|
|
import android.view.WindowInsets;
|
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;
|
|
|
|
|
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;
|
2020-02-07 10:47:03 -08:00
|
|
|
import java.util.function.Function;
|
|
|
|
|
import java.util.function.Supplier;
|
2019-06-04 13:59:43 -07:00
|
|
|
|
2020-02-07 10:47:03 -08:00
|
|
|
/**
|
|
|
|
|
* Class to handle requests from tests
|
|
|
|
|
*/
|
|
|
|
|
@TargetApi(Build.VERSION_CODES.Q)
|
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;
|
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();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Bundle call(String method) {
|
|
|
|
|
final Bundle response = new Bundle();
|
|
|
|
|
switch (method) {
|
|
|
|
|
case TestProtocol.REQUEST_ALL_APPS_TO_OVERVIEW_SWIPE_HEIGHT: {
|
2020-02-07 10:47:03 -08:00
|
|
|
return getLauncherUIProperty(Bundle::putInt, l -> {
|
|
|
|
|
final float progress = LauncherState.OVERVIEW.getVerticalProgress(l)
|
|
|
|
|
- LauncherState.ALL_APPS.getVerticalProgress(l);
|
|
|
|
|
final float distance = l.getAllAppsController().getShiftRange() * progress;
|
|
|
|
|
return (int) distance;
|
|
|
|
|
});
|
2019-05-23 00:50:08 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case TestProtocol.REQUEST_HOME_TO_ALL_APPS_SWIPE_HEIGHT: {
|
2020-02-07 10:47:03 -08:00
|
|
|
return getLauncherUIProperty(Bundle::putInt, l -> {
|
|
|
|
|
final float progress = LauncherState.NORMAL.getVerticalProgress(l)
|
|
|
|
|
- LauncherState.ALL_APPS.getVerticalProgress(l);
|
|
|
|
|
final float distance = l.getAllAppsController().getShiftRange() * progress;
|
|
|
|
|
return (int) distance;
|
|
|
|
|
});
|
2019-05-23 00:50:08 -07:00
|
|
|
}
|
|
|
|
|
|
2019-06-28 12:04:36 -07:00
|
|
|
case TestProtocol.REQUEST_IS_LAUNCHER_INITIALIZED: {
|
2020-02-07 10:47:03 -08:00
|
|
|
return getUIProperty(Bundle::putBoolean, t -> isLauncherInitialized(), () -> true);
|
2019-06-28 12:04:36 -07:00
|
|
|
}
|
|
|
|
|
|
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:
|
2020-02-07 10:47:03 -08:00
|
|
|
return getLauncherUIProperty(Bundle::putBoolean, l -> {
|
|
|
|
|
l.getAppsView().getAppsStore().enableDeferUpdates(DEFER_UPDATES_TEST);
|
|
|
|
|
return true;
|
|
|
|
|
});
|
2019-06-04 13:59:43 -07:00
|
|
|
case TestProtocol.REQUEST_UNFREEZE_APP_LIST:
|
2020-02-07 10:47:03 -08:00
|
|
|
return getLauncherUIProperty(Bundle::putBoolean, l -> {
|
|
|
|
|
l.getAppsView().getAppsStore().disableDeferUpdates(DEFER_UPDATES_TEST);
|
|
|
|
|
return true;
|
|
|
|
|
});
|
2019-06-04 13:59:43 -07:00
|
|
|
|
|
|
|
|
case TestProtocol.REQUEST_APP_LIST_FREEZE_FLAGS: {
|
2020-02-07 10:47:03 -08:00
|
|
|
return getLauncherUIProperty(Bundle::putInt,
|
|
|
|
|
l -> l.getAppsView().getAppsStore().getDeferUpdatesFlags());
|
2019-06-04 13:59:43 -07:00
|
|
|
}
|
2019-08-08 11:49:24 -07:00
|
|
|
|
2019-09-20 09:37:11 -07:00
|
|
|
case TestProtocol.REQUEST_APPS_LIST_SCROLL_Y: {
|
2020-02-07 10:47:03 -08:00
|
|
|
return getLauncherUIProperty(Bundle::putInt,
|
|
|
|
|
l -> l.getAppsView().getActiveRecyclerView().getCurrentScrollY());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case TestProtocol.REQUEST_WINDOW_INSETS: {
|
|
|
|
|
return getUIProperty(Bundle::putParcelable, a -> {
|
|
|
|
|
WindowInsets insets = a.getWindow()
|
|
|
|
|
.getDecorView().getRootWindowInsets();
|
|
|
|
|
return Insets.max(
|
|
|
|
|
insets.getSystemGestureInsets(), insets.getSystemWindowInsets());
|
|
|
|
|
}, this::getCurrentActivity);
|
2019-09-20 09:37:11 -07:00
|
|
|
}
|
|
|
|
|
|
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;
|
|
|
|
|
}
|
2020-03-30 18:03:59 -07:00
|
|
|
|
|
|
|
|
case TestProtocol.REQUEST_MOCK_SENSOR_ROTATION:
|
|
|
|
|
TestProtocol.sDisableSensorRotation = true;
|
|
|
|
|
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
|
|
|
|
2020-02-07 10:47:03 -08:00
|
|
|
protected Activity getCurrentActivity() {
|
|
|
|
|
return Launcher.ACTIVITY_TRACKER.getCreatedActivity();
|
|
|
|
|
}
|
|
|
|
|
|
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);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-07 10:47:03 -08:00
|
|
|
/**
|
|
|
|
|
* Returns the result by getting a Launcher property on UI thread
|
|
|
|
|
*/
|
|
|
|
|
public static <T> Bundle getLauncherUIProperty(
|
|
|
|
|
BundleSetter<T> bundleSetter, Function<Launcher, T> provider) {
|
|
|
|
|
return getUIProperty(bundleSetter, provider, Launcher.ACTIVITY_TRACKER::getCreatedActivity);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Returns the result by getting a generic property on UI thread
|
|
|
|
|
*/
|
|
|
|
|
private static <S, T> Bundle getUIProperty(
|
|
|
|
|
BundleSetter<T> bundleSetter, Function<S, T> provider, Supplier<S> targetSupplier) {
|
|
|
|
|
try {
|
|
|
|
|
return MAIN_EXECUTOR.submit(() -> {
|
|
|
|
|
S target = targetSupplier.get();
|
|
|
|
|
if (target == null) {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
T value = provider.apply(target);
|
|
|
|
|
Bundle response = new Bundle();
|
|
|
|
|
bundleSetter.set(response, TestProtocol.TEST_INFO_RESPONSE_FIELD, value);
|
|
|
|
|
return response;
|
|
|
|
|
}).get();
|
|
|
|
|
} catch (ExecutionException | InterruptedException e) {
|
|
|
|
|
throw new RuntimeException(e);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Generic interface for setting a fiend in bundle
|
|
|
|
|
* @param <T> the type of value being set
|
|
|
|
|
*/
|
|
|
|
|
public interface BundleSetter<T> {
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Sets any generic property to the bundle
|
|
|
|
|
*/
|
|
|
|
|
void set(Bundle b, String key, T value);
|
|
|
|
|
}
|
|
|
|
|
|
2019-10-25 18:12:25 -07:00
|
|
|
// 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
|
|
|
}
|