1 /*
2  * Copyright (C) 2020 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package android.perftests.utils;
18 
19 import static androidx.test.platform.app.InstrumentationRegistry.getInstrumentation;
20 
21 import android.app.KeyguardManager;
22 import android.app.UiAutomation;
23 import android.content.Context;
24 import android.content.Intent;
25 import android.os.ParcelFileDescriptor;
26 import android.os.PowerManager;
27 import android.os.SystemClock;
28 
29 import androidx.test.rule.ActivityTestRule;
30 
31 import org.junit.After;
32 import org.junit.BeforeClass;
33 
34 import java.io.ByteArrayOutputStream;
35 import java.io.File;
36 import java.io.FileInputStream;
37 import java.io.IOException;
38 import java.io.InputStream;
39 import java.util.Objects;
40 import java.util.concurrent.TimeUnit;
41 import java.util.function.Consumer;
42 
43 /** The base class for window related performance tests. */
44 public class WindowPerfTestBase {
45     public static final long NANOS_PER_S = 1000L * 1000 * 1000;
46     public static final long TIME_1_S_IN_NS = 1 * NANOS_PER_S;
47 
48     static boolean sIsProfilingMethod;
49 
50     @BeforeClass
setUpOnce()51     public static void setUpOnce() {
52         final Context context = getInstrumentation().getContext();
53 
54         if (!context.getSystemService(PowerManager.class).isInteractive()
55                 || context.getSystemService(KeyguardManager.class).isKeyguardLocked()) {
56             executeShellCommand("input keyevent KEYCODE_WAKEUP");
57             executeShellCommand("wm dismiss-keyguard");
58         }
59         context.startActivity(new Intent(Intent.ACTION_MAIN)
60                 .addCategory(Intent.CATEGORY_HOME).setFlags(Intent.FLAG_ACTIVITY_NEW_TASK));
61     }
62 
63     @After
tearDown()64     public void tearDown() {
65         // Make sure that profiling is stopped if test fails.
66         if (sIsProfilingMethod) {
67             stopProfiling();
68         }
69     }
70 
getUiAutomation()71     public static UiAutomation getUiAutomation() {
72         return getInstrumentation().getUiAutomation();
73     }
74 
startAsyncAtrace(String tags)75     public static void startAsyncAtrace(String tags) {
76         getUiAutomation().executeShellCommand("atrace --async_start -b 32768 -c " + tags);
77         // Avoid atrace isn't ready immediately.
78         SystemClock.sleep(TimeUnit.NANOSECONDS.toMillis(TIME_1_S_IN_NS));
79     }
80 
stopAsyncAtraceWithStream()81     public static InputStream stopAsyncAtraceWithStream() {
82         return new ParcelFileDescriptor.AutoCloseInputStream(
83                 getUiAutomation().executeShellCommand("atrace --async_stop"));
84     }
85 
86     /** Starts method tracing on system server. */
startProfiling(File basePath, String outFileName)87     public static void startProfiling(File basePath, String outFileName) {
88         if (!basePath.exists()) {
89             executeShellCommand("mkdir -p " + basePath);
90         }
91         final String samplingArg = WindowPerfRunPreconditionBase.sSamplingIntervalUs > 0
92                 ? ("--sampling " + WindowPerfRunPreconditionBase.sSamplingIntervalUs)
93                 : "";
94         executeShellCommand("am profile start " + samplingArg + " system "
95                 + new File(basePath, outFileName));
96         sIsProfilingMethod = true;
97     }
98 
99     /** Stops method tracing of system server. */
stopProfiling()100     public static void stopProfiling() {
101         executeShellCommand("am profile stop system");
102         sIsProfilingMethod = false;
103     }
104 
sIsProfilingMethod()105     public static boolean sIsProfilingMethod() {
106         return sIsProfilingMethod;
107     }
108 
109     /** Returns how many iterations should run with method tracing. */
getProfilingIterations()110     public static int getProfilingIterations() {
111         return WindowPerfRunPreconditionBase.sProfilingIterations;
112     }
113 
114     /**
115      * Executes shell command with reading the output. It may also used to block until the current
116      * command is completed.
117      */
executeShellCommand(String command)118     public static ByteArrayOutputStream executeShellCommand(String command) {
119         final ParcelFileDescriptor pfd = getUiAutomation().executeShellCommand(command);
120         final byte[] buf = new byte[512];
121         final ByteArrayOutputStream bytes = new ByteArrayOutputStream();
122         int bytesRead;
123         try (FileInputStream fis = new ParcelFileDescriptor.AutoCloseInputStream(pfd)) {
124             while ((bytesRead = fis.read(buf)) != -1) {
125                 bytes.write(buf, 0, bytesRead);
126             }
127         } catch (IOException e) {
128             throw new RuntimeException(e);
129         }
130         return bytes;
131     }
132 
runWithShellPermissionIdentity(Runnable runnable)133     public static void runWithShellPermissionIdentity(Runnable runnable) {
134         getUiAutomation().adoptShellPermissionIdentity();
135         try {
136             runnable.run();
137         } finally {
138             getUiAutomation().dropShellPermissionIdentity();
139         }
140     }
141 
142     public static class SettingsSession<T> implements AutoCloseable {
143         private final Consumer<T> mSetter;
144         private final T mOriginalValue;
145         private boolean mChanged;
146 
SettingsSession(T originalValue, Consumer<T> setter)147         public SettingsSession(T originalValue, Consumer<T> setter) {
148             mOriginalValue = originalValue;
149             mSetter = setter;
150         }
151 
set(T value)152         public void set(T value) {
153             if (Objects.equals(value, mOriginalValue)) {
154                 mChanged = false;
155                 return;
156             }
157             mSetter.accept(value);
158             mChanged = true;
159         }
160 
161         @Override
close()162         public void close() {
163             if (mChanged) {
164                 mSetter.accept(mOriginalValue);
165             }
166         }
167     }
168 
169     /**
170      * Provides the {@link PerfTestActivity} with an associated customizable intent.
171      */
172     public static class PerfTestActivityRuleBase extends ActivityTestRule<PerfTestActivity> {
173         protected final Intent mStartIntent =
174                 new Intent(getInstrumentation().getTargetContext(), PerfTestActivity.class);
175 
PerfTestActivityRuleBase()176         public PerfTestActivityRuleBase() {
177             this(false /* launchActivity */);
178         }
179 
PerfTestActivityRuleBase(boolean launchActivity)180         public PerfTestActivityRuleBase(boolean launchActivity) {
181             super(PerfTestActivity.class, false /* initialTouchMode */, launchActivity);
182         }
183 
184         @Override
getActivityIntent()185         public Intent getActivityIntent() {
186             return mStartIntent;
187         }
188 
launchActivity()189         public PerfTestActivity launchActivity() {
190             return launchActivity(mStartIntent);
191         }
192     }
193 }
194