1 /* 2 * Copyright (C) 2018 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 android.annotation.IntDef; 20 import android.app.Activity; 21 import android.app.Instrumentation; 22 import android.os.Bundle; 23 import android.util.ArrayMap; 24 import android.util.Log; 25 26 import com.android.internal.util.ArrayUtils; 27 28 import java.lang.annotation.ElementType; 29 import java.lang.annotation.Retention; 30 import java.lang.annotation.RetentionPolicy; 31 import java.lang.annotation.Target; 32 import java.util.ArrayList; 33 import java.util.Arrays; 34 import java.util.List; 35 import java.util.concurrent.TimeUnit; 36 37 /** 38 * Provides a benchmark framework. 39 * 40 * This differs from BenchmarkState in that rather than the class measuring the the elapsed time, 41 * the test passes in the elapsed time. 42 * 43 * Example usage: 44 * 45 * public void sampleMethod() { 46 * ManualBenchmarkState state = new ManualBenchmarkState(); 47 * 48 * int[] src = new int[] { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 }; 49 * long elapsedTime = 0; 50 * while (state.keepRunning(elapsedTime)) { 51 * long startTime = System.nanoTime(); 52 * int[] dest = new int[src.length]; 53 * System.arraycopy(src, 0, dest, 0, src.length); 54 * elapsedTime = System.nanoTime() - startTime; 55 * } 56 * System.out.println(state.summaryLine()); 57 * } 58 * 59 * Or use the PerfManualStatusReporter TestRule. 60 * 61 * Make sure that the overhead of checking the clock does not noticeably affect the results. 62 */ 63 public final class ManualBenchmarkState { 64 private static final String TAG = ManualBenchmarkState.class.getSimpleName(); 65 66 @Target(ElementType.ANNOTATION_TYPE) 67 @Retention(RetentionPolicy.RUNTIME) 68 public @interface StatsReport { 69 int FLAG_MEDIAN = 0x00000001; 70 int FLAG_MEAN = 0x00000002; 71 int FLAG_MIN = 0x00000004; 72 int FLAG_MAX = 0x00000008; 73 int FLAG_STDDEV = 0x00000010; 74 int FLAG_COEFFICIENT_VAR = 0x00000020; 75 int FLAG_ITERATION = 0x00000040; 76 77 @Retention(RetentionPolicy.RUNTIME) 78 @IntDef(value = { 79 FLAG_MEDIAN, 80 FLAG_MEAN, 81 FLAG_MIN, 82 FLAG_MAX, 83 FLAG_STDDEV, 84 FLAG_COEFFICIENT_VAR, 85 FLAG_ITERATION, 86 }) 87 @interface Flag {} 88 89 /** Defines which type of statistics should output. */ flags()90 @Flag int flags() default -1; 91 /** An array with value 0~100 to provide the percentiles. */ percentiles()92 int[] percentiles() default {}; 93 } 94 95 /** The interface to receive the events of customized iteration. */ 96 public interface CustomizedIterationListener { 97 /** The customized iteration starts. */ onStart(int iteration)98 void onStart(int iteration); 99 100 /** The customized iteration finished. */ onFinished(int iteration)101 void onFinished(int iteration); 102 } 103 104 /** It means the entire {@link StatsReport} is not given. */ 105 private static final int DEFAULT_STATS_REPORT = -2; 106 107 // TODO: Tune these values. 108 // warm-up for duration 109 private static final long WARMUP_DURATION_NS = TimeUnit.SECONDS.toNanos(5); 110 // minimum iterations to warm-up for 111 private static final int WARMUP_MIN_ITERATIONS = 8; 112 113 // target testing for duration 114 private static final long TARGET_TEST_DURATION_NS = TimeUnit.SECONDS.toNanos(16); 115 private static final int MAX_TEST_ITERATIONS = 1000000; 116 private static final int MIN_TEST_ITERATIONS = 10; 117 118 private static final int NOT_STARTED = 0; // The benchmark has not started yet. 119 private static final int WARMUP = 1; // The benchmark is warming up. 120 private static final int RUNNING = 2; // The benchmark is running. 121 private static final int RUNNING_CUSTOMIZED = 3; // Running for customized measurement. 122 private static final int FINISHED = 4; // The benchmark has stopped. 123 124 private int mState = NOT_STARTED; // Current benchmark state. 125 126 private long mWarmupDurationNs = WARMUP_DURATION_NS; 127 private long mTargetTestDurationNs = TARGET_TEST_DURATION_NS; 128 private long mWarmupStartTime = 0; 129 private int mWarmupIterations = 0; 130 131 private int mMaxIterations = 0; 132 133 /** 134 * Additinal iteration that used to apply customized measurement. The result during these 135 * iterations won't be counted into {@link #mStats}. 136 */ 137 private int mMaxCustomizedIterations; 138 private int mCustomizedIterations; 139 private CustomizedIterationListener mCustomizedIterationListener; 140 141 // Individual duration in nano seconds. 142 private ArrayList<Long> mResults = new ArrayList<>(); 143 144 /** @see #addExtraResult(String, long) */ 145 private ArrayMap<String, ArrayList<Long>> mExtraResults; 146 147 private final List<Long> mTmpDurations = Arrays.asList(0L); 148 149 // Statistics. These values will be filled when the benchmark has finished. 150 // The computation needs double precision, but long int is fine for final reporting. 151 private Stats mStats; 152 153 private int mStatsReportFlags = 154 StatsReport.FLAG_MEDIAN | StatsReport.FLAG_MEAN | StatsReport.FLAG_STDDEV; 155 private int[] mStatsReportPercentiles = {90 , 95}; 156 shouldReport(int statsReportFlag)157 private boolean shouldReport(int statsReportFlag) { 158 return (mStatsReportFlags & statsReportFlag) != 0; 159 } 160 configure(ManualBenchmarkTest testAnnotation)161 void configure(ManualBenchmarkTest testAnnotation) { 162 if (testAnnotation == null) { 163 return; 164 } 165 166 final long warmupDurationNs = testAnnotation.warmupDurationNs(); 167 if (warmupDurationNs >= 0) { 168 mWarmupDurationNs = warmupDurationNs; 169 } 170 final long targetTestDurationNs = testAnnotation.targetTestDurationNs(); 171 if (targetTestDurationNs >= 0) { 172 mTargetTestDurationNs = targetTestDurationNs; 173 } 174 final StatsReport statsReport = testAnnotation.statsReport(); 175 if (statsReport != null && statsReport.flags() != DEFAULT_STATS_REPORT) { 176 mStatsReportFlags = statsReport.flags(); 177 mStatsReportPercentiles = statsReport.percentiles(); 178 } 179 } 180 beginBenchmark(long warmupDuration, int iterations)181 private void beginBenchmark(long warmupDuration, int iterations) { 182 mMaxIterations = (int) (mTargetTestDurationNs / (warmupDuration / iterations)); 183 mMaxIterations = Math.min(MAX_TEST_ITERATIONS, 184 Math.max(mMaxIterations, MIN_TEST_ITERATIONS)); 185 mState = RUNNING; 186 } 187 188 /** 189 * Judges whether the benchmark needs more samples. 190 * 191 * For the usage, see class comment. 192 */ keepRunning(long duration)193 public boolean keepRunning(long duration) { 194 if (duration < 0) { 195 throw new RuntimeException("duration is negative: " + duration); 196 } 197 mTmpDurations.set(0, duration); 198 return keepRunning(mTmpDurations); 199 } 200 201 /** 202 * Similar to the {@link #keepRunning(long)} but accepts a list of durations 203 */ keepRunning(List<Long> durations)204 public boolean keepRunning(List<Long> durations) { 205 switch (mState) { 206 case NOT_STARTED: 207 mState = WARMUP; 208 mWarmupStartTime = System.nanoTime(); 209 return true; 210 case WARMUP: { 211 if (ArrayUtils.isEmpty(durations)) { 212 return true; 213 } 214 final long timeSinceStartingWarmup = System.nanoTime() - mWarmupStartTime; 215 mWarmupIterations += durations.size(); 216 if (mWarmupIterations >= WARMUP_MIN_ITERATIONS 217 && timeSinceStartingWarmup >= mWarmupDurationNs) { 218 beginBenchmark(timeSinceStartingWarmup, mWarmupIterations); 219 } 220 return true; 221 } 222 case RUNNING: { 223 if (ArrayUtils.isEmpty(durations)) { 224 return true; 225 } 226 mResults.addAll(durations); 227 final boolean keepRunning = mResults.size() < mMaxIterations; 228 if (!keepRunning) { 229 mStats = new Stats(mResults); 230 if (mMaxCustomizedIterations > 0 && mCustomizedIterationListener != null) { 231 mState = RUNNING_CUSTOMIZED; 232 mCustomizedIterationListener.onStart(mCustomizedIterations); 233 return true; 234 } 235 mState = FINISHED; 236 } 237 return keepRunning; 238 } 239 case RUNNING_CUSTOMIZED: { 240 mCustomizedIterationListener.onFinished(mCustomizedIterations); 241 mCustomizedIterations++; 242 if (mCustomizedIterations >= mMaxCustomizedIterations) { 243 mState = FINISHED; 244 return false; 245 } 246 mCustomizedIterationListener.onStart(mCustomizedIterations); 247 return true; 248 } 249 case FINISHED: 250 throw new IllegalStateException("The benchmark has finished."); 251 default: 252 throw new IllegalStateException("The benchmark is in an unknown state."); 253 } 254 } 255 256 /** 257 * @return {@code true} if the benchmark is in warmup state. It can be used to skip the 258 * operations or measurements that are unnecessary while the test isn't running the 259 * actual benchmark. 260 */ isWarmingUp()261 public boolean isWarmingUp() { 262 return mState == WARMUP; 263 } 264 265 /** 266 * This is used to run the benchmark with more information by enabling some debug mechanism but 267 * we don't want to account the special runs (slower) in the stats report. 268 */ setCustomizedIterations(int iterations, CustomizedIterationListener listener)269 public void setCustomizedIterations(int iterations, CustomizedIterationListener listener) { 270 mMaxCustomizedIterations = iterations; 271 mCustomizedIterationListener = listener; 272 } 273 274 /** 275 * Adds additional result while this benchmark isn't warming up or running in customized state. 276 * It is used when a sequence of operations is executed consecutively, the duration of each 277 * operation can also be recorded. 278 */ addExtraResult(String key, long duration)279 public void addExtraResult(String key, long duration) { 280 if (isWarmingUp() || mState == RUNNING_CUSTOMIZED) { 281 return; 282 } 283 if (mExtraResults == null) { 284 mExtraResults = new ArrayMap<>(); 285 } 286 mExtraResults.computeIfAbsent(key, k -> new ArrayList<>()).add(duration); 287 } 288 summaryLine(String key, Stats stats, ArrayList<Long> results)289 private static String summaryLine(String key, Stats stats, ArrayList<Long> results) { 290 final StringBuilder sb = new StringBuilder(key); 291 sb.append(" Summary: "); 292 sb.append("median=").append(stats.getMedian()).append("ns, "); 293 sb.append("mean=").append(stats.getMean()).append("ns, "); 294 sb.append("min=").append(stats.getMin()).append("ns, "); 295 sb.append("max=").append(stats.getMax()).append("ns, "); 296 sb.append("sigma=").append(stats.getStandardDeviation()).append(", "); 297 sb.append("iteration=").append(results.size()).append(", "); 298 sb.append("values="); 299 if (results.size() > 100) { 300 sb.append(results.subList(0, 100)).append(" ..."); 301 } else { 302 sb.append(results); 303 } 304 return sb.toString(); 305 } 306 fillStatus(Bundle status, String key, Stats stats)307 private void fillStatus(Bundle status, String key, Stats stats) { 308 if (shouldReport(StatsReport.FLAG_ITERATION)) { 309 status.putLong(key + "_iteration", stats.getSize()); 310 } 311 if (shouldReport(StatsReport.FLAG_MEDIAN)) { 312 status.putLong(key + "_median (ns)", stats.getMedian()); 313 } 314 if (shouldReport(StatsReport.FLAG_MEAN)) { 315 status.putLong(key + "_mean (ns)", Math.round(stats.getMean())); 316 } 317 if (shouldReport(StatsReport.FLAG_MIN)) { 318 status.putLong(key + "_min (ns)", stats.getMin()); 319 } 320 if (shouldReport(StatsReport.FLAG_MAX)) { 321 status.putLong(key + "_max (ns)", stats.getMax()); 322 } 323 if (mStatsReportPercentiles != null) { 324 for (int percentile : mStatsReportPercentiles) { 325 status.putLong(key + "_percentile" + percentile + " (ns)", 326 stats.getPercentile(percentile)); 327 } 328 } 329 if (shouldReport(StatsReport.FLAG_STDDEV)) { 330 status.putLong(key + "_stddev (ns)", Math.round(stats.getStandardDeviation())); 331 } 332 if (shouldReport(StatsReport.FLAG_COEFFICIENT_VAR)) { 333 status.putLong(key + "_cv", 334 Math.round((100 * stats.getStandardDeviation() / stats.getMean()))); 335 } 336 } 337 sendFullStatusReport(Instrumentation instrumentation, String key)338 public void sendFullStatusReport(Instrumentation instrumentation, String key) { 339 if (mState != FINISHED) { 340 throw new IllegalStateException("The benchmark hasn't finished"); 341 } 342 Log.i(TAG, summaryLine(key, mStats, mResults)); 343 final Bundle status = new Bundle(); 344 fillStatus(status, key, mStats); 345 if (mExtraResults != null) { 346 for (int i = 0; i < mExtraResults.size(); i++) { 347 final String subKey = key + "_" + mExtraResults.keyAt(i); 348 final ArrayList<Long> results = mExtraResults.valueAt(i); 349 final Stats stats = new Stats(results); 350 Log.i(TAG, summaryLine(subKey, stats, results)); 351 fillStatus(status, subKey, stats); 352 } 353 } 354 instrumentation.sendStatus(Activity.RESULT_OK, status); 355 } 356 357 /** The annotation to customize the test, e.g. the duration of warm-up and target test. */ 358 @Target(ElementType.METHOD) 359 @Retention(RetentionPolicy.RUNTIME) 360 public @interface ManualBenchmarkTest { warmupDurationNs()361 long warmupDurationNs() default -1; targetTestDurationNs()362 long targetTestDurationNs() default -1; statsReport()363 StatsReport statsReport() default @StatsReport(flags = DEFAULT_STATS_REPORT); 364 } 365 } 366